OSDN Git Service

am b3968bbc: Disable thumbnail view mode for most visited page.
[android-x86/packages-apps-Browser.git] / src / com / android / browser / BrowserBookmarksPage.java
1 /*
2  * Copyright (C) 2006 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16
17 package com.android.browser;
18
19 import android.app.Activity;
20 import android.app.AlertDialog;
21 import android.content.DialogInterface;
22 import android.content.Intent;
23 import android.content.SharedPreferences;
24 import android.content.SharedPreferences.Editor;
25 import android.graphics.Bitmap;
26 import android.graphics.BitmapFactory;
27 import android.graphics.Canvas;
28 import android.graphics.Color;
29 import android.graphics.Paint;
30 import android.graphics.Path;
31 import android.graphics.PorterDuff;
32 import android.graphics.PorterDuffXfermode;
33 import android.graphics.RectF;
34 import android.net.Uri;
35 import android.os.Bundle;
36 import android.os.Handler;
37 import android.os.Message;
38 import android.os.ServiceManager;
39 import android.provider.Browser;
40 import android.text.IClipboard;
41 import android.util.Log;
42 import android.view.ContextMenu;
43 import android.view.KeyEvent;
44 import android.view.LayoutInflater;
45 import android.view.Menu;
46 import android.view.MenuInflater;
47 import android.view.MenuItem;
48 import android.view.View;
49 import android.view.ViewGroup;
50 import android.view.ViewGroup.LayoutParams;
51 import android.view.ViewStub;
52 import android.view.ContextMenu.ContextMenuInfo;
53 import android.widget.AdapterView;
54 import android.widget.GridView;
55 import android.widget.ListView;
56 import android.widget.Toast;
57
58 /*package*/ enum BookmarkViewMode { NONE, GRID, LIST }
59 /**
60  *  View showing the user's bookmarks in the browser.
61  */
62 public class BrowserBookmarksPage extends Activity implements 
63         View.OnCreateContextMenuListener {
64
65     private BookmarkViewMode        mViewMode = BookmarkViewMode.NONE;
66     private GridView                mGridPage;
67     private View                    mVerticalList;
68     private BrowserBookmarksAdapter mBookmarksAdapter;
69     private static final int        BOOKMARKS_SAVE = 1;
70     private boolean                 mDisableNewWindow;
71     private BookmarkItem            mContextHeader;
72     private AddNewBookmark          mAddHeader;
73     private boolean                 mCanceled = false;
74     private boolean                 mCreateShortcut;
75     private boolean                 mMostVisited;
76     private View                    mEmptyView;
77     // XXX: There is no public string defining this intent so if Home changes
78     // the value, we have to update this string.
79     private static final String     INSTALL_SHORTCUT =
80             "com.android.launcher.action.INSTALL_SHORTCUT";
81     
82     private final static String LOGTAG = "browser";
83     private final static String PREF_BOOKMARK_VIEW_MODE = "pref_bookmark_view_mode";
84     private final static String PREF_MOST_VISITED_VIEW_MODE = "pref_most_visited_view_mode";
85
86     @Override
87     public boolean onContextItemSelected(MenuItem item) {
88         // It is possible that the view has been canceled when we get to
89         // this point as back has a higher priority 
90         if (mCanceled) {
91             return true;
92         }
93         AdapterView.AdapterContextMenuInfo i = 
94             (AdapterView.AdapterContextMenuInfo)item.getMenuInfo();
95         // If we have no menu info, we can't tell which item was selected.
96         if (i == null) {
97             return true;
98         }
99         
100         switch (item.getItemId()) {
101         case R.id.new_context_menu_id:
102             saveCurrentPage();
103             break;
104         case R.id.open_context_menu_id:
105             loadUrl(i.position);
106             break;
107         case R.id.edit_context_menu_id:
108             editBookmark(i.position);
109             break;
110         case R.id.shortcut_context_menu_id:
111             final Intent send = createShortcutIntent(i.position);
112             send.setAction(INSTALL_SHORTCUT);
113             sendBroadcast(send);
114             break;
115         case R.id.delete_context_menu_id:
116             if (mMostVisited) {
117                 Browser.deleteFromHistory(getContentResolver(),
118                         getUrl(i.position));
119                 refreshList();
120             } else {
121                 displayRemoveBookmarkDialog(i.position);
122             }
123             break;
124         case R.id.new_window_context_menu_id:
125             openInNewWindow(i.position);
126             break;
127         case R.id.share_link_context_menu_id:
128             Browser.sendString(BrowserBookmarksPage.this, getUrl(i.position),
129                     getText(R.string.choosertitle_sharevia).toString());
130             break;
131         case R.id.copy_url_context_menu_id:
132             copy(getUrl(i.position));
133             break;
134         case R.id.homepage_context_menu_id:
135             BrowserSettings.getInstance().setHomePage(this,
136                     getUrl(i.position));
137             Toast.makeText(this, R.string.homepage_set,
138                     Toast.LENGTH_LONG).show();
139             break;
140         // Only for the Most visited page
141         case R.id.save_to_bookmarks_menu_id:
142             boolean isBookmark;
143             String name;
144             String url;
145             if (mViewMode == BookmarkViewMode.GRID) {
146                 isBookmark = mBookmarksAdapter.getIsBookmark(i.position);
147                 name = mBookmarksAdapter.getTitle(i.position);
148                 url = mBookmarksAdapter.getUrl(i.position);
149             } else {
150                 HistoryItem historyItem = ((HistoryItem) i.targetView);
151                 isBookmark = historyItem.isBookmark();
152                 name = historyItem.getName();
153                 url = historyItem.getUrl();
154             }
155             // If the site is bookmarked, the item becomes remove from
156             // bookmarks.
157             if (isBookmark) {
158                 Bookmarks.removeFromBookmarks(this, getContentResolver(), url, name);
159             } else {
160                 Browser.saveBookmark(this, name, url);
161             }
162             break;
163         default:
164             return super.onContextItemSelected(item);
165         }
166         return true;
167     }
168
169     @Override
170     public void onCreateContextMenu(ContextMenu menu, View v,
171                 ContextMenuInfo menuInfo) {
172             AdapterView.AdapterContextMenuInfo i = 
173                     (AdapterView.AdapterContextMenuInfo) menuInfo;
174
175             MenuInflater inflater = getMenuInflater();
176             if (mMostVisited) {
177                 inflater.inflate(R.menu.historycontext, menu);
178             } else {
179                 inflater.inflate(R.menu.bookmarkscontext, menu);
180             }
181
182             if (0 == i.position && !mMostVisited) {
183                 menu.setGroupVisible(R.id.CONTEXT_MENU, false);
184                 if (mAddHeader == null) {
185                     mAddHeader = new AddNewBookmark(BrowserBookmarksPage.this);
186                 } else if (mAddHeader.getParent() != null) {
187                     ((ViewGroup) mAddHeader.getParent()).
188                             removeView(mAddHeader);
189                 }
190                 mAddHeader.setUrl(getIntent().getStringExtra("url"));
191                 menu.setHeaderView(mAddHeader);
192                 return;
193             }
194             if (mMostVisited) {
195                 if ((mViewMode == BookmarkViewMode.LIST
196                         && ((HistoryItem) i.targetView).isBookmark())
197                         || mBookmarksAdapter.getIsBookmark(i.position)) {
198                     MenuItem item = menu.findItem(
199                             R.id.save_to_bookmarks_menu_id);
200                     item.setTitle(R.string.remove_from_bookmarks);
201                 }
202             } else {
203                 // The historycontext menu has no ADD_MENU group.
204                 menu.setGroupVisible(R.id.ADD_MENU, false);
205             }
206             if (mDisableNewWindow) {
207                 menu.findItem(R.id.new_window_context_menu_id).setVisible(
208                         false);
209             }
210             if (mContextHeader == null) {
211                 mContextHeader = new BookmarkItem(BrowserBookmarksPage.this);
212             } else if (mContextHeader.getParent() != null) {
213                 ((ViewGroup) mContextHeader.getParent()).
214                         removeView(mContextHeader);
215             }
216             if (mViewMode == BookmarkViewMode.GRID) {
217                 mBookmarksAdapter.populateBookmarkItem(mContextHeader,
218                         i.position);
219             } else {
220                 BookmarkItem b = (BookmarkItem) i.targetView;
221                 b.copyTo(mContextHeader);
222             }
223             menu.setHeaderView(mContextHeader);
224         }
225
226     /**
227      *  Create a new BrowserBookmarksPage.
228      */  
229     @Override
230     protected void onCreate(Bundle icicle) {
231         super.onCreate(icicle);
232
233         if (Intent.ACTION_CREATE_SHORTCUT.equals(getIntent().getAction())) {
234             mCreateShortcut = true;
235         }
236         mDisableNewWindow = getIntent().getBooleanExtra("disable_new_window",
237                 false);
238         mMostVisited = getIntent().getBooleanExtra("mostVisited", false);
239
240         if (mCreateShortcut) {
241             setTitle(R.string.browser_bookmarks_page_bookmarks_text);
242         }
243         mBookmarksAdapter = new BrowserBookmarksAdapter(this,
244                         getIntent().getStringExtra("url"),
245                         getIntent().getStringExtra("title"),
246                         (Bitmap) getIntent().getParcelableExtra("thumbnail"),
247                         mCreateShortcut,
248                         mMostVisited);
249
250         setContentView(R.layout.empty_history);
251         mEmptyView = findViewById(R.id.empty_view);
252         mEmptyView.setVisibility(View.GONE);
253
254         SharedPreferences p = getPreferences(MODE_PRIVATE);
255
256         // See if the user has set a preference for the view mode of their
257         // bookmarks. Otherwise default to grid mode.
258         BookmarkViewMode preference = BookmarkViewMode.NONE;
259         if (mMostVisited) {
260             // For the most visited page, only use list mode.
261             preference = BookmarkViewMode.LIST;
262         } else {
263             preference = BookmarkViewMode.values()[p.getInt(
264                     PREF_BOOKMARK_VIEW_MODE, BookmarkViewMode.GRID.ordinal())];
265         }
266         switchViewMode(preference);
267     }
268
269     /**
270      *  Set the ContentView to be either the grid of thumbnails or the vertical
271      *  list.
272      */
273     private void switchViewMode(BookmarkViewMode gridMode) {
274         if (mViewMode == gridMode) {
275             return;
276         }
277
278         mViewMode = gridMode;
279
280         // Update the preferences to make the new view mode sticky.
281         Editor ed = getPreferences(MODE_PRIVATE).edit();
282         if (mMostVisited) {
283             ed.putInt(PREF_MOST_VISITED_VIEW_MODE, mViewMode.ordinal());
284         } else {
285             ed.putInt(PREF_BOOKMARK_VIEW_MODE, mViewMode.ordinal());
286         }
287         ed.commit();
288
289         mBookmarksAdapter.switchViewMode(gridMode);
290         if (mViewMode == BookmarkViewMode.GRID) {
291             if (mGridPage == null) {
292                 mGridPage = new GridView(this);
293                 mGridPage.setAdapter(mBookmarksAdapter);
294                 mGridPage.setOnItemClickListener(mListener);
295                 mGridPage.setNumColumns(GridView.AUTO_FIT);
296                 mGridPage.setColumnWidth(
297                         BrowserActivity.getDesiredThumbnailWidth(this));
298                 mGridPage.setFocusable(true);
299                 mGridPage.setFocusableInTouchMode(true);
300                 mGridPage.setSelector(android.R.drawable.gallery_thumb);
301                 float density = getResources().getDisplayMetrics().density;
302                 mGridPage.setVerticalSpacing((int) (14 * density));
303                 mGridPage.setHorizontalSpacing((int) (8 * density));
304                 mGridPage.setStretchMode(GridView.STRETCH_SPACING);
305                 mGridPage.setScrollBarStyle(View.SCROLLBARS_INSIDE_INSET);
306                 mGridPage.setDrawSelectorOnTop(true);
307                 if (mMostVisited) {
308                     mGridPage.setEmptyView(mEmptyView);
309                 }
310                 if (!mCreateShortcut) {
311                     mGridPage.setOnCreateContextMenuListener(this);
312                 }
313             }
314             addContentView(mGridPage, FULL_SCREEN_PARAMS);
315             if (mVerticalList != null) {
316                 ViewGroup parent = (ViewGroup) mVerticalList.getParent();
317                 if (parent != null) {
318                     parent.removeView(mVerticalList);
319                 }
320             }
321         } else {
322             if (null == mVerticalList) {
323                 ListView listView = new ListView(this);
324                 listView.setAdapter(mBookmarksAdapter);
325                 listView.setDrawSelectorOnTop(false);
326                 listView.setVerticalScrollBarEnabled(true);
327                 listView.setOnItemClickListener(mListener);
328                 if (mMostVisited) {
329                     listView.setEmptyView(mEmptyView);
330                 }
331                 if (!mCreateShortcut) {
332                     listView.setOnCreateContextMenuListener(this);
333                 }
334                 mVerticalList = listView;
335             }
336             addContentView(mVerticalList, FULL_SCREEN_PARAMS);
337             if (mGridPage != null) {
338                 ViewGroup parent = (ViewGroup) mGridPage.getParent();
339                 if (parent != null) {
340                     parent.removeView(mGridPage);
341                 }
342             }
343         }
344     }
345
346     private static final ViewGroup.LayoutParams FULL_SCREEN_PARAMS
347             = new ViewGroup.LayoutParams(
348             ViewGroup.LayoutParams.FILL_PARENT,
349             ViewGroup.LayoutParams.FILL_PARENT);
350
351     private static final int SAVE_CURRENT_PAGE = 1000;
352     private final Handler mHandler = new Handler() {
353         @Override
354         public void handleMessage(Message msg) {
355             if (msg.what == SAVE_CURRENT_PAGE) {
356                 saveCurrentPage();
357             }
358         }
359     };
360
361     private AdapterView.OnItemClickListener mListener = new AdapterView.OnItemClickListener() {
362         public void onItemClick(AdapterView parent, View v, int position, long id) {
363             // It is possible that the view has been canceled when we get to
364             // this point as back has a higher priority 
365             if (mCanceled) {
366                 android.util.Log.e(LOGTAG, "item clicked when dismissing");
367                 return;
368             }
369             if (!mCreateShortcut) {
370                 if (0 == position && !mMostVisited) {
371                     // XXX: Work-around for a framework issue.
372                     mHandler.sendEmptyMessage(SAVE_CURRENT_PAGE);
373                 } else {
374                     loadUrl(position);
375                 }
376             } else {
377                 final Intent intent = createShortcutIntent(position);
378                 setResultToParent(RESULT_OK, intent);
379                 finish();
380             }
381         }
382     };
383
384     private Intent createShortcutIntent(int position) {
385         String url = getUrl(position);
386         String title = getBookmarkTitle(position);
387         Bitmap touchIcon = getTouchIcon(position);
388
389         final Intent i = new Intent();
390         final Intent shortcutIntent = new Intent(Intent.ACTION_VIEW,
391                 Uri.parse(url));
392         long urlHash = url.hashCode();
393         long uniqueId = (urlHash << 32) | shortcutIntent.hashCode();
394         shortcutIntent.putExtra(Browser.EXTRA_APPLICATION_ID,
395                 Long.toString(uniqueId));
396         i.putExtra(Intent.EXTRA_SHORTCUT_INTENT, shortcutIntent);
397         i.putExtra(Intent.EXTRA_SHORTCUT_NAME, title);
398         // Use the apple-touch-icon if available
399         if (touchIcon != null) {
400             // Make a copy so we can modify the pixels.
401             Bitmap copy = touchIcon.copy(Bitmap.Config.ARGB_8888, true);
402             Canvas canvas = new Canvas(copy);
403
404             // Construct a path from a round rect. This will allow drawing with
405             // an inverse fill so we can punch a hole using the round rect.
406             Path path = new Path();
407             path.setFillType(Path.FillType.INVERSE_WINDING);
408             RectF rect = new RectF(0, 0, touchIcon.getWidth(),
409                     touchIcon.getHeight());
410             rect.inset(1, 1);
411             path.addRoundRect(rect, 8f, 8f, Path.Direction.CW);
412
413             // Construct a paint that clears the outside of the rectangle and
414             // draw.
415             Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
416             paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
417             canvas.drawPath(path, paint);
418
419             i.putExtra(Intent.EXTRA_SHORTCUT_ICON, copy);
420         } else {
421             Bitmap favicon = getFavicon(position);
422             if (favicon == null) {
423                 i.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE,
424                         Intent.ShortcutIconResource.fromContext(
425                                 BrowserBookmarksPage.this,
426                                 R.drawable.ic_launcher_shortcut_browser_bookmark));
427             } else {
428                 Bitmap icon = BitmapFactory.decodeResource(getResources(),
429                         R.drawable.ic_launcher_shortcut_browser_bookmark);
430
431                 // Make a copy of the regular icon so we can modify the pixels.
432                 Bitmap copy = icon.copy(Bitmap.Config.ARGB_8888, true);
433                 Canvas canvas = new Canvas(copy);
434
435                 // Make a Paint for the white background rectangle and for
436                 // filtering the favicon.
437                 Paint p = new Paint(Paint.ANTI_ALIAS_FLAG
438                         | Paint.FILTER_BITMAP_FLAG);
439                 p.setStyle(Paint.Style.FILL_AND_STROKE);
440                 p.setColor(Color.WHITE);
441
442                 float density = getResources().getDisplayMetrics().density;
443                 // Create a rectangle that is slightly wider than the favicon
444                 final float iconSize = 16 * density; // 16x16 favicon
445                 final float padding = 2; // white padding around icon
446                 final float rectSize = iconSize + 2 * padding;
447                 final float y = icon.getHeight() - rectSize;
448                 RectF r = new RectF(0, y, rectSize, y + rectSize);
449
450                 // Draw a white rounded rectangle behind the favicon
451                 canvas.drawRoundRect(r, 2, 2, p);
452
453                 // Draw the favicon in the same rectangle as the rounded
454                 // rectangle but inset by the padding
455                 // (results in a 16x16 favicon).
456                 r.inset(padding, padding);
457                 canvas.drawBitmap(favicon, null, r, p);
458                 i.putExtra(Intent.EXTRA_SHORTCUT_ICON, copy);
459             }
460         }
461         // Do not allow duplicate items
462         i.putExtra("duplicate", false);
463         return i;
464     }
465
466     private void saveCurrentPage() {
467         Intent i = new Intent(BrowserBookmarksPage.this,
468                 AddBookmarkPage.class);
469         i.putExtras(getIntent());
470         startActivityForResult(i, BOOKMARKS_SAVE);
471     }
472
473     private void loadUrl(int position) {
474         Intent intent = (new Intent()).setAction(getUrl(position));
475         setResultToParent(RESULT_OK, intent);
476         finish();
477     }
478
479     @Override
480     public boolean onCreateOptionsMenu(Menu menu) {
481         boolean result = super.onCreateOptionsMenu(menu);
482         if (!mCreateShortcut && !mMostVisited) {
483             MenuInflater inflater = getMenuInflater();
484             inflater.inflate(R.menu.bookmarks, menu);
485             return true;
486         }
487         return result;
488     }
489
490     @Override
491     public boolean onPrepareOptionsMenu(Menu menu) {
492         boolean result = super.onPrepareOptionsMenu(menu);
493         if (mCreateShortcut || mMostVisited
494                 || mBookmarksAdapter.getCount() == 0) {
495             // No need to show the menu if there are no items.
496             return result;
497         }
498         MenuItem switchItem = menu.findItem(R.id.switch_mode_menu_id);
499         int titleResId;
500         int iconResId;
501         if (mViewMode == BookmarkViewMode.GRID) {
502             titleResId = R.string.switch_to_list;
503             iconResId = R.drawable.ic_menu_list;
504         } else {
505             titleResId = R.string.switch_to_thumbnails;
506             iconResId = R.drawable.ic_menu_thumbnail;
507         }
508         switchItem.setTitle(titleResId);
509         switchItem.setIcon(iconResId);
510         return true;
511     }
512
513     @Override
514     public boolean onOptionsItemSelected(MenuItem item) {
515         switch (item.getItemId()) {
516         case R.id.new_context_menu_id:
517             saveCurrentPage();
518             break;
519
520         case R.id.switch_mode_menu_id:
521             if (mViewMode == BookmarkViewMode.GRID) {
522                 switchViewMode(BookmarkViewMode.LIST);
523             } else {
524                 switchViewMode(BookmarkViewMode.GRID);
525             }
526             break;
527
528         default:
529             return super.onOptionsItemSelected(item);
530         }
531         return true;
532     }
533
534     private void openInNewWindow(int position) {
535         Bundle b = new Bundle();
536         b.putBoolean("new_window", true);
537         setResultToParent(RESULT_OK,
538                 (new Intent()).setAction(getUrl(position)).putExtras(b));
539
540         finish();
541     }
542     
543
544     private void editBookmark(int position) {
545         Intent intent = new Intent(BrowserBookmarksPage.this, 
546             AddBookmarkPage.class);
547         intent.putExtra("bookmark", getRow(position));
548         startActivityForResult(intent, BOOKMARKS_SAVE);
549     }
550
551     @Override
552     protected void onActivityResult(int requestCode, int resultCode,
553                                     Intent data) {
554         switch(requestCode) {
555             case BOOKMARKS_SAVE:
556                 if (resultCode == RESULT_OK) {
557                     Bundle extras;
558                     if (data != null && (extras = data.getExtras()) != null) {
559                         // If there are extras, then we need to save
560                         // the edited bookmark. This is done in updateRow()
561                         String title = extras.getString("title");
562                         String url = extras.getString("url");
563                         if (title != null && url != null) {
564                             mBookmarksAdapter.updateRow(extras);
565                         }
566                     } else {
567                         // extras == null then a new bookmark was added to
568                         // the database.
569                         refreshList();
570                     }
571                 }
572                 break;
573             default:
574                 break;
575         }
576     }
577     
578     private void displayRemoveBookmarkDialog(int position) {
579         // Put up a dialog asking if the user really wants to
580         // delete the bookmark
581         final int deletePos = position;
582         new AlertDialog.Builder(this)
583                 .setTitle(R.string.delete_bookmark)
584                 .setIcon(android.R.drawable.ic_dialog_alert)
585                 .setMessage(getText(R.string.delete_bookmark_warning).toString().replace(
586                         "%s", getBookmarkTitle(deletePos)))
587                 .setPositiveButton(R.string.ok, 
588                         new DialogInterface.OnClickListener() {
589                             public void onClick(DialogInterface dialog, int whichButton) {
590                                 deleteBookmark(deletePos);
591                             }
592                         })
593                 .setNegativeButton(R.string.cancel, null)
594                 .show();
595     }
596
597     /**
598      *  Refresh the shown list after the database has changed.
599      */
600     private void refreshList() {
601         mBookmarksAdapter.refreshList();
602     }
603     
604     /**
605      *  Return a hashmap representing the currently highlighted row.
606      */
607     public Bundle getRow(int position) {
608         return mBookmarksAdapter.getRow(position);
609     }
610
611     /**
612      *  Return the url of the currently highlighted row.
613      */
614     public String getUrl(int position) {
615         return mBookmarksAdapter.getUrl(position);
616     }
617
618     /**
619      * Return the favicon of the currently highlighted row.
620      */
621     public Bitmap getFavicon(int position) {
622         return mBookmarksAdapter.getFavicon(position);
623     }
624
625     private Bitmap getTouchIcon(int position) {
626         return mBookmarksAdapter.getTouchIcon(position);
627     }
628
629     private void copy(CharSequence text) {
630         try {
631             IClipboard clip = IClipboard.Stub.asInterface(ServiceManager.getService("clipboard"));
632             if (clip != null) {
633                 clip.setClipboardText(text);
634             }
635         } catch (android.os.RemoteException e) {
636             Log.e(LOGTAG, "Copy failed", e);
637         }
638     }
639     
640     public String getBookmarkTitle(int position) {
641         return mBookmarksAdapter.getTitle(position);
642     }
643
644     /**
645      *  Delete the currently highlighted row.
646      */
647     public void deleteBookmark(int position) {
648         mBookmarksAdapter.deleteRow(position);
649     }
650
651     @Override
652     public void onBackPressed() {
653         setResultToParent(RESULT_CANCELED, null);
654         mCanceled = true;
655         super.onBackPressed();
656     }
657
658     // This Activity is generally a sub-Activity of CombinedHistoryActivity. In
659     // that situation, we need to pass our result code up to our parent.
660     // However, if someone calls this Activity directly, then this has no
661     // parent, and it needs to set it on itself.
662     private void setResultToParent(int resultCode, Intent data) {
663         Activity a = getParent() == null ? this : getParent();
664         a.setResult(resultCode, data);
665     }
666 }