OSDN Git Service

am 9f53534e: (-s ours) Import revised translations. DO NOT MERGE
[android-x86/packages-apps-Browser.git] / src / com / android / browser / TabControl.java
1 /*
2  * Copyright (C) 2007 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.content.Context;
20 import android.graphics.Picture;
21 import android.net.http.SslError;
22 import android.os.Bundle;
23 import android.os.Message;
24 import android.util.Log;
25 import android.view.Gravity;
26 import android.view.LayoutInflater;
27 import android.view.View;
28 import android.view.ViewGroup;
29 import android.view.View.OnClickListener;
30 import android.webkit.HttpAuthHandler;
31 import android.webkit.JsPromptResult;
32 import android.webkit.JsResult;
33 import android.webkit.SslErrorHandler;
34 import android.webkit.WebBackForwardList;
35 import android.webkit.WebChromeClient;
36 import android.webkit.WebHistoryItem;
37 import android.webkit.WebView;
38 import android.webkit.WebViewClient;
39 import android.widget.FrameLayout;
40 import android.widget.ImageButton;
41
42 import java.io.File;
43 import java.io.FileInputStream;
44 import java.util.ArrayList;
45 import java.util.Vector;
46
47 class TabControl {
48     // Log Tag
49     private static final String LOGTAG = "TabControl";
50     // Maximum number of tabs.
51     static final int MAX_TABS = 8;
52     // Static instance of an empty callback.
53     private static final WebViewClient mEmptyClient =
54             new WebViewClient();
55     // Instance of BackgroundChromeClient for background tabs.
56     private final BackgroundChromeClient mBackgroundChromeClient =
57             new BackgroundChromeClient();
58     // Private array of WebViews that are used as tabs.
59     private ArrayList<Tab> mTabs = new ArrayList<Tab>(MAX_TABS);
60     // Queue of most recently viewed tabs.
61     private ArrayList<Tab> mTabQueue = new ArrayList<Tab>(MAX_TABS);
62     // Current position in mTabs.
63     private int mCurrentTab = -1;
64     // A private instance of BrowserActivity to interface with when adding and
65     // switching between tabs.
66     private final BrowserActivity mActivity;
67     // Inflation service for making subwindows.
68     private final LayoutInflater mInflateService;
69     // Subclass of WebViewClient used in subwindows to notify the main
70     // WebViewClient of certain WebView activities.
71     private class SubWindowClient extends WebViewClient {
72         // The main WebViewClient.
73         private final WebViewClient mClient;
74
75         SubWindowClient(WebViewClient client) {
76             mClient = client;
77         }
78         @Override
79         public void doUpdateVisitedHistory(WebView view, String url,
80                 boolean isReload) {
81             mClient.doUpdateVisitedHistory(view, url, isReload);
82         }
83         @Override
84         public boolean shouldOverrideUrlLoading(WebView view, String url) {
85             return mClient.shouldOverrideUrlLoading(view, url);
86         }
87         @Override
88         public void onReceivedSslError(WebView view, SslErrorHandler handler,
89                 SslError error) {
90             mClient.onReceivedSslError(view, handler, error);
91         }
92         @Override
93         public void onReceivedHttpAuthRequest(WebView view,
94                 HttpAuthHandler handler, String host, String realm) {
95             mClient.onReceivedHttpAuthRequest(view, handler, host, realm);
96         }
97         @Override
98         public void onFormResubmission(WebView view, Message dontResend,
99                 Message resend) {
100             mClient.onFormResubmission(view, dontResend, resend);
101         }
102         @Override
103         public void onReceivedError(WebView view, int errorCode,
104                 String description, String failingUrl) {
105             mClient.onReceivedError(view, errorCode, description, failingUrl);
106         }
107     }
108     // Subclass of WebChromeClient to display javascript dialogs.
109     private class SubWindowChromeClient extends WebChromeClient {
110         // This subwindow's tab.
111         private final Tab mTab;
112         // The main WebChromeClient.
113         private final WebChromeClient mClient;
114
115         SubWindowChromeClient(Tab t, WebChromeClient client) {
116             mTab = t;
117             mClient = client;
118         }
119         @Override
120         public void onProgressChanged(WebView view, int newProgress) {
121             mClient.onProgressChanged(view, newProgress);
122         }
123         @Override
124         public boolean onCreateWindow(WebView view, boolean dialog,
125                 boolean userGesture, android.os.Message resultMsg) {
126             return mClient.onCreateWindow(view, dialog, userGesture, resultMsg);
127         }
128         @Override
129         public void onCloseWindow(WebView window) {
130             if (Browser.DEBUG && window != mTab.mSubView) {
131                 throw new AssertionError("Can't close the window");
132             }
133             mActivity.dismissSubWindow(mTab);
134         }
135     }
136     // Background WebChromeClient for focusing tabs
137     private class BackgroundChromeClient extends WebChromeClient {
138         @Override
139         public void onRequestFocus(WebView view) {
140             Tab t = getTabFromView(view);
141             if (t != getCurrentTab()) {
142                 mActivity.showTab(t);
143             }
144         }
145     }
146
147     // Extra saved information for displaying the tab in the picker.
148     public static class PickerData {
149         String  mUrl;
150         String  mTitle;
151         float   mScale;
152         int     mScrollX;
153         int     mScrollY;
154         int     mWidth;
155         Picture mPicture;
156         // This can be null. When a new picture comes in, this view should be
157         // invalidated to show the new picture.
158         FakeWebView mFakeWebView;
159     }
160
161     /**
162      * Private class for maintaining Tabs with a main WebView and a subwindow.
163      */
164     public class Tab implements WebView.PictureListener {
165         // Main WebView
166         private WebView mMainView;
167         // Subwindow WebView
168         private WebView mSubView;
169         // Subwindow container
170         private View mSubViewContainer;
171         // Subwindow callback
172         private SubWindowClient mSubViewClient;
173         // Subwindow chrome callback
174         private SubWindowChromeClient mSubViewChromeClient;
175         // Saved bundle for when we are running low on memory. It contains the
176         // information needed to restore the WebView if the user goes back to
177         // the tab.
178         private Bundle mSavedState;
179         // Data used when displaying the tab in the picker.
180         private PickerData mPickerData;
181
182         // Parent Tab. This is the Tab that created this Tab, or null
183         // if the Tab was created by the UI
184         private Tab mParentTab;
185         // Tab that constructed by this Tab. This is used when this
186         // Tab is destroyed, it clears all mParentTab values in the 
187         // children.
188         private Vector<Tab> mChildTabs;
189
190         private Boolean mCloseOnExit;
191         // Application identifier used to find tabs that another application
192         // wants to reuse.
193         private String mAppId;
194         // Keep the original url around to avoid killing the old WebView if the
195         // url has not changed.
196         private String mOriginalUrl;
197
198         private ErrorConsoleView mErrorConsole;
199
200         // Construct a new tab
201         private Tab(WebView w, boolean closeOnExit, String appId, String url) {
202             mMainView = w;
203             mCloseOnExit = closeOnExit;
204             mAppId = appId;
205             mOriginalUrl = url;
206         }
207
208         /**
209          * Return the top window of this tab; either the subwindow if it is not
210          * null or the main window.
211          * @return The top window of this tab.
212          */
213         public WebView getTopWindow() {
214             if (mSubView != null) {
215                 return mSubView;
216             }
217             return mMainView;
218         }
219
220         /**
221          * Return the main window of this tab. Note: if a tab is freed in the
222          * background, this can return null. It is only guaranteed to be 
223          * non-null for the current tab.
224          * @return The main WebView of this tab.
225          */
226         public WebView getWebView() {
227             return mMainView;
228         }
229
230         /**
231          * Return the subwindow of this tab or null if there is no subwindow.
232          * @return The subwindow of this tab or null.
233          */
234         public WebView getSubWebView() {
235             return mSubView;
236         }
237
238         /**
239          * Return the subwindow container of this tab or null if there is no
240          * subwindow.
241          * @return The subwindow's container View.
242          */
243         public View getSubWebViewContainer() {
244             return mSubViewContainer;
245         }
246
247         /**
248          * Get the url of this tab.  Valid after calling populatePickerData, but
249          * before calling wipePickerData, or if the webview has been destroyed.
250          * 
251          * @return The WebView's url or null.
252          */
253         public String getUrl() {
254             if (mPickerData != null) {
255                 return mPickerData.mUrl;
256             }
257             return null;
258         }
259
260         /**
261          * Get the title of this tab.  Valid after calling populatePickerData, 
262          * but before calling wipePickerData, or if the webview has been 
263          * destroyed.  If the url has no title, use the url instead.
264          * 
265          * @return The WebView's title (or url) or null.
266          */
267         public String getTitle() {
268             if (mPickerData != null) {
269                 return mPickerData.mTitle;
270             }
271             return null;
272         }
273
274         /**
275          * Returns the picker data.
276          */
277         public PickerData getPickerData() {
278             return mPickerData;
279         }
280
281         private void setParentTab(Tab parent) {
282             mParentTab = parent;
283             // This tab may have been freed due to low memory. If that is the
284             // case, the parent tab index is already saved. If we are changing
285             // that index (most likely due to removing the parent tab) we must
286             // update the parent tab index in the saved Bundle.
287             if (mSavedState != null) {
288                 if (parent == null) {
289                     mSavedState.remove(PARENTTAB);
290                 } else {
291                     mSavedState.putInt(PARENTTAB, getTabIndex(parent));
292                 }
293             }
294         }
295         
296         /**
297          * When a Tab is created through the content of another Tab, then 
298          * we associate the Tabs. 
299          * @param child the Tab that was created from this Tab
300          */
301         public void addChildTab(Tab child) {
302             if (mChildTabs == null) {
303                 mChildTabs = new Vector<Tab>();
304             }
305             mChildTabs.add(child);
306             child.setParentTab(this);
307         }
308         
309         private void removeFromTree() {
310             // detach the children
311             if (mChildTabs != null) {
312                 for(Tab t : mChildTabs) {
313                     t.setParentTab(null);
314                 }
315             }
316             
317             // Find myself in my parent list
318             if (mParentTab != null) {
319                 mParentTab.mChildTabs.remove(this);
320             }
321         }
322         
323         /**
324          * If this Tab was created through another Tab, then this method
325          * returns that Tab.
326          * @return the Tab parent or null
327          */
328         public Tab getParentTab() {
329             return mParentTab;
330         }
331
332         /**
333          * Return whether this tab should be closed when it is backing out of
334          * the first page.
335          * @return TRUE if this tab should be closed when exit.
336          */
337         public boolean closeOnExit() {
338             return mCloseOnExit;
339         }
340
341         public void onNewPicture(WebView view, Picture p) {
342             if (mPickerData == null) {
343                 return;
344             }
345
346             mPickerData.mPicture = p;
347             // Tell the FakeWebView to redraw.
348             if (mPickerData.mFakeWebView != null) {
349                 mPickerData.mFakeWebView.invalidate();
350             }
351         }
352     };
353
354     // Directory to store thumbnails for each WebView.
355     private final File mThumbnailDir;
356
357     /**
358      * Construct a new TabControl object that interfaces with the given
359      * BrowserActivity instance.
360      * @param activity A BrowserActivity instance that TabControl will interface
361      *                 with.
362      */
363     TabControl(BrowserActivity activity) {
364         mActivity = activity;
365         mInflateService =
366                 ((LayoutInflater) activity.getSystemService(
367                         Context.LAYOUT_INFLATER_SERVICE));
368         mThumbnailDir = activity.getDir("thumbnails", 0);
369     }
370
371     File getThumbnailDir() {
372         return mThumbnailDir;
373     }
374
375     BrowserActivity getBrowserActivity() {
376         return mActivity;
377     }
378
379     /**
380      * Return the current tab's main WebView. This will always return the main
381      * WebView for a given tab and not a subwindow.
382      * @return The current tab's WebView.
383      */
384     WebView getCurrentWebView() {
385         Tab t = getTab(mCurrentTab);
386         if (t == null) {
387             return null;
388         }
389         return t.mMainView;
390     }
391
392     /**
393      * Return the current tab's error console. Creates the console if createIfNEcessary
394      * is true and we haven't already created the console.
395      * @param createIfNecessary Flag to indicate if the console should be created if it has
396      *                          not been already.
397      * @return The current tab's error console, or null if one has not been created and
398      *         createIfNecessary is false.
399      */
400     ErrorConsoleView getCurrentErrorConsole(boolean createIfNecessary) {
401         Tab t = getTab(mCurrentTab);
402         if (t == null) {
403             return null;
404         }
405
406         if (createIfNecessary && t.mErrorConsole == null) {
407             t.mErrorConsole = new ErrorConsoleView(mActivity);
408             t.mErrorConsole.setWebView(t.mMainView);
409         }
410
411         return t.mErrorConsole;
412     }
413
414     /**
415      * Return the current tab's top-level WebView. This can return a subwindow
416      * if one exists.
417      * @return The top-level WebView of the current tab.
418      */
419     WebView getCurrentTopWebView() {
420         Tab t = getTab(mCurrentTab);
421         if (t == null) {
422             return null;
423         }
424         return t.mSubView != null ? t.mSubView : t.mMainView;
425     }
426
427     /**
428      * Return the current tab's subwindow if it exists.
429      * @return The subwindow of the current tab or null if it doesn't exist.
430      */
431     WebView getCurrentSubWindow() {
432         Tab t = getTab(mCurrentTab);
433         if (t == null) {
434             return null;
435         }
436         return t.mSubView;
437     }
438
439     /**
440      * Return the tab at the specified index.
441      * @return The Tab for the specified index or null if the tab does not
442      *         exist.
443      */
444     Tab getTab(int index) {
445         if (index >= 0 && index < mTabs.size()) {
446             return mTabs.get(index);
447         }
448         return null;
449     }
450
451     /**
452      * Return the current tab.
453      * @return The current tab.
454      */
455     Tab getCurrentTab() {
456         return getTab(mCurrentTab);
457     }
458
459     /**
460      * Return the current tab index.
461      * @return The current tab index
462      */
463     int getCurrentIndex() {
464         return mCurrentTab;
465     }
466     
467     /**
468      * Given a Tab, find it's index
469      * @param Tab to find
470      * @return index of Tab or -1 if not found
471      */
472     int getTabIndex(Tab tab) {
473         if (tab == null) {
474             return -1;
475         }
476         return mTabs.indexOf(tab);
477     }
478
479     /**
480      * Create a new tab.
481      * @return The newly createTab or null if we have reached the maximum
482      *         number of open tabs.
483      */
484     Tab createNewTab(boolean closeOnExit, String appId, String url) {
485         int size = mTabs.size();
486         // Return false if we have maxed out on tabs
487         if (MAX_TABS == size) {
488             return null;
489         }
490         final WebView w = createNewWebView();
491         // Create a new tab and add it to the tab list
492         Tab t = new Tab(w, closeOnExit, appId, url);
493         mTabs.add(t);
494         // Initially put the tab in the background.
495         putTabInBackground(t);
496         return t;
497     }
498
499     /**
500      * Create a new tab with default values for closeOnExit(false),
501      * appId(null), and url(null).
502      */
503     Tab createNewTab() {
504         return createNewTab(false, null, null);
505     }
506
507     /**
508      * Remove the tab from the list. If the tab is the current tab shown, the
509      * last created tab will be shown.
510      * @param t The tab to be removed.
511      */
512     boolean removeTab(Tab t) {
513         if (t == null) {
514             return false;
515         }
516         // Only remove the tab if it is the current one.
517         if (getCurrentTab() == t) {
518             putTabInBackground(t);
519         }
520
521         // Only destroy the WebView if it still exists.
522         if (t.mMainView != null) {
523             // Take down the sub window.
524             dismissSubWindow(t);
525             // Remove the WebView's settings from the BrowserSettings list of
526             // observers.
527             BrowserSettings.getInstance().deleteObserver(
528                     t.mMainView.getSettings());
529             // Destroy the main view and subview
530             t.mMainView.destroy();
531             t.mMainView = null;
532         }
533         // clear it's references to parent and children
534         t.removeFromTree();
535         
536         // Remove it from our list of tabs.
537         mTabs.remove(t);
538
539         // The tab indices have shifted, update all the saved state so we point
540         // to the correct index.
541         for (Tab tab : mTabs) {
542             if (tab.mChildTabs != null) {
543                 for (Tab child : tab.mChildTabs) {
544                     child.setParentTab(tab);
545                 }
546             }
547         }
548
549
550         // This tab may have been pushed in to the background and then closed.
551         // If the saved state contains a picture file, delete the file.
552         if (t.mSavedState != null) {
553             if (t.mSavedState.containsKey(CURRPICTURE)) {
554                 new File(t.mSavedState.getString(CURRPICTURE)).delete();
555             }
556         }
557
558         // Remove it from the queue of viewed tabs.
559         mTabQueue.remove(t);
560         mCurrentTab = -1;
561         return true;
562     }
563
564     /**
565      * Clear the back/forward list for all the current tabs.
566      */
567     void clearHistory() {
568         int size = getTabCount();
569         for (int i = 0; i < size; i++) {
570             Tab t = mTabs.get(i);
571             // TODO: if a tab is freed due to low memory, its history is not
572             // cleared here.
573             if (t.mMainView != null) {
574                 t.mMainView.clearHistory();
575             }
576             if (t.mSubView != null) {
577                 t.mSubView.clearHistory();
578             }
579         }
580     }
581
582     /**
583      * Destroy all the tabs and subwindows
584      */
585     void destroy() {
586         BrowserSettings s = BrowserSettings.getInstance();
587         for (Tab t : mTabs) {
588             if (t.mMainView != null) {
589                 dismissSubWindow(t);
590                 s.deleteObserver(t.mMainView.getSettings());
591                 t.mMainView.destroy();
592                 t.mMainView = null;
593             }
594         }
595         mTabs.clear();
596         mTabQueue.clear();
597     }
598
599     /**
600      * Returns the number of tabs created.
601      * @return The number of tabs created.
602      */
603     int getTabCount() {
604         return mTabs.size();
605     }
606
607     // Used for saving and restoring each Tab
608     private static final String WEBVIEW = "webview";
609     private static final String NUMTABS = "numTabs";
610     private static final String CURRTAB = "currentTab";
611     private static final String CURRURL = "currentUrl";
612     private static final String CURRTITLE = "currentTitle";
613     private static final String CURRWIDTH = "currentWidth";
614     private static final String CURRPICTURE = "currentPicture";
615     private static final String CLOSEONEXIT = "closeonexit";
616     private static final String PARENTTAB = "parentTab";
617     private static final String APPID = "appid";
618     private static final String ORIGINALURL = "originalUrl";
619
620     /**
621      * Save the state of all the Tabs.
622      * @param outState The Bundle to save the state to.
623      */
624     void saveState(Bundle outState) {
625         final int numTabs = getTabCount();
626         outState.putInt(NUMTABS, numTabs);
627         final int index = getCurrentIndex();
628         outState.putInt(CURRTAB, (index >= 0 && index < numTabs) ? index : 0);
629         for (int i = 0; i < numTabs; i++) {
630             final Tab t = getTab(i);
631             if (saveState(t)) {
632                 outState.putBundle(WEBVIEW + i, t.mSavedState);
633             }
634         }
635     }
636
637     /**
638      * Restore the state of all the tabs.
639      * @param inState The saved state of all the tabs.
640      * @return True if there were previous tabs that were restored. False if
641      *         there was no saved state or restoring the state failed.
642      */
643     boolean restoreState(Bundle inState) {
644         final int numTabs = (inState == null)
645                 ? -1 : inState.getInt(NUMTABS, -1);
646         if (numTabs == -1) {
647             return false;
648         } else {
649             final int currentTab = inState.getInt(CURRTAB, -1);
650             for (int i = 0; i < numTabs; i++) {
651                 if (i == currentTab) {
652                     Tab t = createNewTab();
653                     // Me must set the current tab before restoring the state
654                     // so that all the client classes are set.
655                     setCurrentTab(t);
656                     if (!restoreState(inState.getBundle(WEBVIEW + i), t)) {
657                         Log.w(LOGTAG, "Fail in restoreState, load home page.");
658                         t.mMainView.loadUrl(BrowserSettings.getInstance()
659                                 .getHomePage());
660                     }
661                 } else {
662                     // Create a new tab and don't restore the state yet, add it
663                     // to the tab list
664                     Tab t = new Tab(null, false, null, null);
665                     t.mSavedState = inState.getBundle(WEBVIEW + i);
666                     if (t.mSavedState != null) {
667                         populatePickerDataFromSavedState(t);
668                         // Need to maintain the app id and original url so we
669                         // can possibly reuse this tab.
670                         t.mAppId = t.mSavedState.getString(APPID);
671                         t.mOriginalUrl = t.mSavedState.getString(ORIGINALURL);
672                     }
673                     mTabs.add(t);
674                     mTabQueue.add(t);
675                 }
676             }
677             // Rebuild the tree of tabs. Do this after all tabs have been
678             // created/restored so that the parent tab exists.
679             for (int i = 0; i < numTabs; i++) {
680                 final Bundle b = inState.getBundle(WEBVIEW + i);
681                 final Tab t = getTab(i);
682                 if (b != null && t != null) {
683                     final int parentIndex = b.getInt(PARENTTAB, -1);
684                     if (parentIndex != -1) {
685                         final Tab parent = getTab(parentIndex);
686                         if (parent != null) {
687                             parent.addChildTab(t);
688                         }
689                     }
690                 }
691             }
692         }
693         return true;
694     }
695
696     /**
697      * Free the memory in this order, 1) free the background tab; 2) free the
698      * WebView cache;
699      */
700     void freeMemory() {
701         if (getTabCount() == 0) return;
702
703         // free the least frequently used background tab
704         Tab t = getLeastUsedTab(getCurrentTab().getParentTab());
705         if (t != null) {
706             Log.w(LOGTAG, "Free a tab in the browser");
707             freeTab(t);
708             // force a gc
709             System.gc();
710             return;
711         }
712
713         // free the WebView's unused memory (this includes the cache)
714         Log.w(LOGTAG, "Free WebView's unused memory and cache");
715         WebView view = getCurrentWebView();
716         if (view != null) {
717             view.freeMemory();
718         }
719         // force a gc
720         System.gc();
721     }
722
723     private Tab getLeastUsedTab(Tab currentParent) {
724         // Don't do anything if we only have 1 tab.
725         if (getTabCount() == 1) {
726             return null;
727         }
728
729         // Rip through the queue starting at the beginning and teardown the
730         // next available tab.
731         Tab t = null;
732         int i = 0;
733         final int queueSize = mTabQueue.size();
734         if (queueSize == 0) {
735             return null;
736         }
737         do {
738             t = mTabQueue.get(i++);
739         } while (i < queueSize
740                 && ((t != null && t.mMainView == null) || t == currentParent));
741
742         // Don't do anything if the last remaining tab is the current one or if
743         // the last tab has been freed already.
744         if (t == getCurrentTab() || t.mMainView == null) {
745             return null;
746         }
747
748         return t;
749     }
750
751     private void freeTab(Tab t) {
752         // Store the WebView's state.
753         saveState(t);
754
755         // Tear down the tab.
756         dismissSubWindow(t);
757         // Remove the WebView's settings from the BrowserSettings list of
758         // observers.
759         BrowserSettings.getInstance().deleteObserver(t.mMainView.getSettings());
760         t.mMainView.destroy();
761         t.mMainView = null;
762     }
763
764     /**
765      * Create a new subwindow unless a subwindow already exists.
766      * @return True if a new subwindow was created. False if one already exists.
767      */
768     void createSubWindow() {
769         Tab t = getTab(mCurrentTab);
770         if (t != null && t.mSubView == null) {
771             final View v = mInflateService.inflate(R.layout.browser_subwindow, null);
772             final WebView w = (WebView) v.findViewById(R.id.webview);
773             w.setMapTrackballToArrowKeys(false); // use trackball directly
774             final SubWindowClient subClient =
775                     new SubWindowClient(mActivity.getWebViewClient());
776             final SubWindowChromeClient subChromeClient =
777                     new SubWindowChromeClient(t,
778                             mActivity.getWebChromeClient());
779             w.setWebViewClient(subClient);
780             w.setWebChromeClient(subChromeClient);
781             w.setDownloadListener(mActivity);
782             w.setOnCreateContextMenuListener(mActivity);
783             final BrowserSettings s = BrowserSettings.getInstance();
784             s.addObserver(w.getSettings()).update(s, null);
785             t.mSubView = w;
786             t.mSubViewClient = subClient;
787             t.mSubViewChromeClient = subChromeClient;
788             // FIXME: I really hate having to know the name of the view
789             // containing the webview.
790             t.mSubViewContainer = v.findViewById(R.id.subwindow_container);
791             final ImageButton cancel =
792                     (ImageButton) v.findViewById(R.id.subwindow_close);
793             cancel.setOnClickListener(new OnClickListener() {
794                     public void onClick(View v) {
795                         subChromeClient.onCloseWindow(w);
796                     }
797                 });
798         }
799     }
800
801     /**
802      * Show the tab that contains the given WebView.
803      * @param view The WebView used to find the tab.
804      */
805     Tab getTabFromView(WebView view) {
806         final int size = getTabCount();
807         for (int i = 0; i < size; i++) {
808             final Tab t = getTab(i);
809             if (t.mSubView == view || t.mMainView == view) {
810                 return t;
811             }
812         }
813         return null;
814     }
815
816     /**
817      * Return the tab with the matching application id.
818      * @param id The application identifier.
819      */
820     Tab getTabFromId(String id) {
821         if (id == null) {
822             return null;
823         }
824         final int size = getTabCount();
825         for (int i = 0; i < size; i++) {
826             final Tab t = getTab(i);
827             if (id.equals(t.mAppId)) {
828                 return t;
829             }
830         }
831         return null;
832     }
833
834     // This method checks if a non-app tab (one created within the browser)
835     // matches the given url.
836     private boolean tabMatchesUrl(Tab t, String url) {
837         if (t.mAppId != null) {
838             return false;
839         } else if (t.mMainView == null) {
840             return false;
841         } else if (url.equals(t.mMainView.getUrl()) ||
842                 url.equals(t.mMainView.getOriginalUrl())) {
843             return true;
844         }
845         return false;
846     }
847
848     /**
849      * Return the tab that has no app id associated with it and the url of the
850      * tab matches the given url.
851      * @param url The url to search for.
852      */
853     Tab findUnusedTabWithUrl(String url) {
854         if (url == null) {
855             return null;
856         }
857         // Check the current tab first.
858         Tab t = getCurrentTab();
859         if (t != null && tabMatchesUrl(t, url)) {
860             return t;
861         }
862         // Now check all the rest.
863         final int size = getTabCount();
864         for (int i = 0; i < size; i++) {
865             t = getTab(i);
866             if (tabMatchesUrl(t, url)) {
867                 return t;
868             }
869         }
870         return null;
871     }
872
873     /**
874      * Recreate the main WebView of the given tab. Returns true if the WebView
875      * was deleted.
876      */
877     boolean recreateWebView(Tab t, String url) {
878         final WebView w = t.mMainView;
879         if (w != null) {
880             if (url != null && url.equals(t.mOriginalUrl)) {
881                 // The original url matches the current url. Just go back to the
882                 // first history item so we can load it faster than if we
883                 // rebuilt the WebView.
884                 final WebBackForwardList list = w.copyBackForwardList();
885                 if (list != null) {
886                     w.goBackOrForward(-list.getCurrentIndex());
887                     w.clearHistory(); // maintains the current page.
888                     return false;
889                 }
890             }
891             // Remove the settings object from the global settings and destroy
892             // the WebView.
893             BrowserSettings.getInstance().deleteObserver(
894                     t.mMainView.getSettings());
895             t.mMainView.destroy();
896         }
897         // Create a new WebView. If this tab is the current tab, we need to put
898         // back all the clients so force it to be the current tab.
899         t.mMainView = createNewWebView();
900         if (getCurrentTab() == t) {
901             setCurrentTab(t, true);
902         }
903         // Clear the saved state except for the app id and close-on-exit
904         // values.
905         t.mSavedState = null;
906         t.mPickerData = null;
907         // Save the new url in order to avoid deleting the WebView.
908         t.mOriginalUrl = url;
909         return true;
910     }
911
912     /**
913      * Creates a new WebView and registers it with the global settings.
914      */
915     private WebView createNewWebView() {
916         // Create a new WebView
917         WebView w = new WebView(mActivity);
918         w.setMapTrackballToArrowKeys(false); // use trackball directly
919         // Enable the built-in zoom
920         w.getSettings().setBuiltInZoomControls(true);
921         // Add this WebView to the settings observer list and update the
922         // settings
923         final BrowserSettings s = BrowserSettings.getInstance();
924         s.addObserver(w.getSettings()).update(s, null);
925         return w;
926     }
927
928     /**
929      * Put the current tab in the background and set newTab as the current tab.
930      * @param newTab The new tab. If newTab is null, the current tab is not
931      *               set.
932      */
933     boolean setCurrentTab(Tab newTab) {
934         return setCurrentTab(newTab, false);
935     }
936
937     /*package*/ void pauseCurrentTab() {
938         Tab t = getCurrentTab();
939         if (t != null) {
940             t.mMainView.onPause();
941             if (t.mSubView != null) {
942                 t.mSubView.onPause();
943             }
944         }
945     }
946
947     /*package*/ void resumeCurrentTab() {
948         Tab t = getCurrentTab();
949         if (t != null) {
950             t.mMainView.onResume();
951             if (t.mSubView != null) {
952                 t.mSubView.onResume();
953             }
954         }
955     }
956
957     private void putViewInForeground(WebView v, WebViewClient vc,
958                                      WebChromeClient cc) {
959         v.setWebViewClient(vc);
960         v.setWebChromeClient(cc);
961         v.setOnCreateContextMenuListener(mActivity);
962         v.setDownloadListener(mActivity);
963         v.onResume();
964     }
965
966     private void putViewInBackground(WebView v) {
967         // Set an empty callback so that default actions are not triggered.
968         v.setWebViewClient(mEmptyClient);
969         v.setWebChromeClient(mBackgroundChromeClient);
970         v.setOnCreateContextMenuListener(null);
971         // Leave the DownloadManager attached so that downloads can start in
972         // a non-active window. This can happen when going to a site that does
973         // a redirect after a period of time. The user could have switched to
974         // another tab while waiting for the download to start.
975         v.setDownloadListener(mActivity);
976         v.onPause();
977     }
978
979     /**
980      * If force is true, this method skips the check for newTab == current.
981      */
982     private boolean setCurrentTab(Tab newTab, boolean force) {
983         Tab current = getTab(mCurrentTab);
984         if (current == newTab && !force) {
985             return true;
986         }
987         if (current != null) {
988             // Remove the current WebView and the container of the subwindow
989             putTabInBackground(current);
990         }
991
992         if (newTab == null) {
993             return false;
994         }
995
996         // Move the newTab to the end of the queue
997         int index = mTabQueue.indexOf(newTab);
998         if (index != -1) {
999             mTabQueue.remove(index);
1000         }
1001         mTabQueue.add(newTab);
1002
1003         WebView mainView;
1004
1005         // Display the new current tab
1006         mCurrentTab = mTabs.indexOf(newTab);
1007         mainView = newTab.mMainView;
1008         boolean needRestore = (mainView == null);
1009         if (needRestore) {
1010             // Same work as in createNewTab() except don't do new Tab()
1011             newTab.mMainView = mainView = createNewWebView();
1012         }
1013         putViewInForeground(mainView, mActivity.getWebViewClient(),
1014                             mActivity.getWebChromeClient());
1015         // Add the subwindow if it exists
1016         if (newTab.mSubViewContainer != null) {
1017             putViewInForeground(newTab.mSubView, newTab.mSubViewClient,
1018                                 newTab.mSubViewChromeClient);
1019         }
1020         if (needRestore) {
1021             // Have to finish setCurrentTab work before calling restoreState
1022             if (!restoreState(newTab.mSavedState, newTab)) {
1023                 mainView.loadUrl(BrowserSettings.getInstance().getHomePage());
1024             }
1025         }
1026         return true;
1027     }
1028
1029     /*
1030      * Put the tab in the background using all the empty/background clients.
1031      */
1032     private void putTabInBackground(Tab t) {
1033         putViewInBackground(t.mMainView);
1034         if (t.mSubView != null) {
1035             putViewInBackground(t.mSubView);
1036         }
1037     }
1038
1039     /*
1040      * Dismiss the subwindow for the given tab.
1041      */
1042     void dismissSubWindow(Tab t) {
1043         if (t != null && t.mSubView != null) {
1044             BrowserSettings.getInstance().deleteObserver(
1045                     t.mSubView.getSettings());
1046             t.mSubView.destroy();
1047             t.mSubView = null;
1048             t.mSubViewContainer = null;
1049         }
1050     }
1051
1052     /**
1053      * Ensure that Tab t has data to display in the tab picker.
1054      * @param  t   Tab to populate.
1055      */
1056     /* package */ void populatePickerData(Tab t) {
1057         if (t == null) {
1058             return;
1059         }
1060
1061         // mMainView == null indicates that the tab has been freed.
1062         if (t.mMainView == null) {
1063             populatePickerDataFromSavedState(t);
1064             return;
1065         }
1066
1067         // FIXME: The only place we cared about subwindow was for 
1068         // bookmarking (i.e. not when saving state). Was this deliberate?
1069         final WebBackForwardList list = t.mMainView.copyBackForwardList();
1070         final WebHistoryItem item =
1071                 list != null ? list.getCurrentItem() : null;
1072         populatePickerData(t, item);
1073
1074         // This method is only called during the tab picker creation. At this
1075         // point we need to listen for new pictures since the WebView is still
1076         // active.
1077         final WebView w = t.getTopWindow();
1078         w.setPictureListener(t);
1079         // Capture the picture here instead of populatePickerData since it can
1080         // be called when saving the state of a tab.
1081         t.mPickerData.mPicture = w.capturePicture();
1082     }
1083
1084     // Create the PickerData and populate it using the saved state of the tab.
1085     private void populatePickerDataFromSavedState(Tab t) {
1086         if (t.mSavedState == null) {
1087             return;
1088         }
1089
1090         final PickerData data = new PickerData();
1091         final Bundle state = t.mSavedState;
1092         data.mUrl = state.getString(CURRURL);
1093         data.mTitle = state.getString(CURRTITLE);
1094         data.mWidth = state.getInt(CURRWIDTH, 0);
1095         // XXX: These keys are from WebView.savePicture so if they change, this
1096         // will break.
1097         data.mScale = state.getFloat("scale", 1.0f);
1098         data.mScrollX = state.getInt("scrollX", 0);
1099         data.mScrollY = state.getInt("scrollY", 0);
1100
1101         if (state.containsKey(CURRPICTURE)) {
1102             final File f = new File(t.mSavedState.getString(CURRPICTURE));
1103             try {
1104                 final FileInputStream in = new FileInputStream(f);
1105                 data.mPicture = Picture.createFromStream(in);
1106                 in.close();
1107             } catch (Exception ex) {
1108                 // Ignore any problems with inflating the picture. We just
1109                 // won't draw anything.
1110             }
1111         }
1112
1113         // Set the tab's picker data.
1114         t.mPickerData = data;
1115     }
1116
1117     // Populate the picker data using the given history item and the current
1118     // top WebView.
1119     private void populatePickerData(Tab t, WebHistoryItem item) {
1120         final PickerData data = new PickerData();
1121         if (item != null) {
1122             data.mUrl = item.getUrl();
1123             data.mTitle = item.getTitle();
1124             if (data.mTitle == null) {
1125                 data.mTitle = data.mUrl;
1126             }
1127         }
1128         // We want to display the top window in the tab picker but use the url
1129         // and title of the main window.
1130         final WebView w = t.getTopWindow();
1131         data.mWidth = w.getWidth();
1132         data.mScale = w.getScale();
1133         data.mScrollX = w.getScrollX();
1134         data.mScrollY = w.getScrollY();
1135
1136         // Remember the old picture if possible.
1137         if (t.mPickerData != null) {
1138             data.mPicture = t.mPickerData.mPicture;
1139         }
1140         t.mPickerData = data;
1141     }
1142     
1143     /**
1144      * Clean up the data for all tabs.
1145      */
1146     /* package */ void wipeAllPickerData() {
1147         int size = getTabCount();
1148         for (int i = 0; i < size; i++) {
1149             final Tab t = getTab(i);
1150             if (t != null && t.mSavedState == null) {
1151                 t.mPickerData = null;
1152             }
1153             if (t.mMainView != null) {
1154                 // Clear the picture listeners.
1155                 t.mMainView.setPictureListener(null);
1156                 if (t.mSubView != null) {
1157                     t.mSubView.setPictureListener(null);
1158                 }
1159             }
1160         }
1161     }
1162
1163     /*
1164      * Save the state for an individual tab.
1165      */
1166     private boolean saveState(Tab t) {
1167         if (t != null) {
1168             final WebView w = t.mMainView;
1169             // If the WebView is null it means we ran low on memory and we
1170             // already stored the saved state in mSavedState.
1171             if (w == null) {
1172                 return true;
1173             }
1174             final Bundle b = new Bundle();
1175             final WebBackForwardList list = w.saveState(b);
1176             if (list != null) {
1177                 final File f = new File(mThumbnailDir, w.hashCode()
1178                         + "_pic.save");
1179                 if (w.savePicture(b, f)) {
1180                     b.putString(CURRPICTURE, f.getPath());
1181                 }
1182             }
1183
1184             // Store some extra info for displaying the tab in the picker.
1185             final WebHistoryItem item =
1186                     list != null ? list.getCurrentItem() : null;
1187             populatePickerData(t, item);
1188
1189             // XXX: WebView.savePicture stores the scale and scroll positions
1190             // in the bundle so we don't have to do it here.
1191             final PickerData data = t.mPickerData;
1192             if (data.mUrl != null) {
1193                 b.putString(CURRURL, data.mUrl);
1194             }
1195             if (data.mTitle != null) {
1196                 b.putString(CURRTITLE, data.mTitle);
1197             }
1198             b.putInt(CURRWIDTH, data.mWidth);
1199             b.putBoolean(CLOSEONEXIT, t.mCloseOnExit);
1200             if (t.mAppId != null) {
1201                 b.putString(APPID, t.mAppId);
1202             }
1203             if (t.mOriginalUrl != null) {
1204                 b.putString(ORIGINALURL, t.mOriginalUrl);
1205             }
1206
1207             // Remember the parent tab so the relationship can be restored.
1208             if (t.mParentTab != null) {
1209                 b.putInt(PARENTTAB, getTabIndex(t.mParentTab));
1210             }
1211
1212             // Remember the saved state.
1213             t.mSavedState = b;
1214             return true;
1215         }
1216         return false;
1217     }
1218
1219     /*
1220      * Restore the state of the tab.
1221      */
1222     private boolean restoreState(Bundle b, Tab t) {
1223         if (b == null) {
1224             return false;
1225         }
1226         // Restore the internal state even if the WebView fails to restore.
1227         // This will maintain the app id, original url and close-on-exit values.
1228         t.mSavedState = null;
1229         t.mPickerData = null;
1230         t.mCloseOnExit = b.getBoolean(CLOSEONEXIT);
1231         t.mAppId = b.getString(APPID);
1232         t.mOriginalUrl = b.getString(ORIGINALURL);
1233
1234         final WebView w = t.mMainView;
1235         final WebBackForwardList list = w.restoreState(b);
1236         if (list == null) {
1237             return false;
1238         }
1239         if (b.containsKey(CURRPICTURE)) {
1240             final File f = new File(b.getString(CURRPICTURE));
1241             w.restorePicture(b, f);
1242             f.delete();
1243         }
1244         return true;
1245     }
1246 }