blob: c5085b3bb048ae6e51c80e72b4c0893ee3221647 [file] [log] [blame]
The Android Open Source Project0c908882009-03-03 19:32:16 -08001/*
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
17package com.android.browser;
18
19import android.content.Context;
Patrick Scott2ed6edb2009-04-22 10:07:45 -040020import android.graphics.Picture;
The Android Open Source Project0c908882009-03-03 19:32:16 -080021import android.net.http.SslError;
22import android.os.Bundle;
23import android.os.Message;
The Android Open Source Project0c908882009-03-03 19:32:16 -080024import android.util.Log;
25import android.view.Gravity;
26import android.view.LayoutInflater;
27import android.view.View;
28import android.view.ViewGroup;
29import android.view.View.OnClickListener;
30import android.webkit.HttpAuthHandler;
31import android.webkit.JsPromptResult;
32import android.webkit.JsResult;
33import android.webkit.SslErrorHandler;
34import android.webkit.WebBackForwardList;
35import android.webkit.WebChromeClient;
36import android.webkit.WebHistoryItem;
37import android.webkit.WebView;
38import android.webkit.WebViewClient;
39import android.widget.FrameLayout;
40import android.widget.ImageButton;
41
42import java.io.File;
Patrick Scott2ed6edb2009-04-22 10:07:45 -040043import java.io.FileInputStream;
The Android Open Source Project0c908882009-03-03 19:32:16 -080044import java.util.ArrayList;
45import java.util.Vector;
46
47class 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) {
Dave Bort31a6d1c2009-04-13 15:56:49 -0700130 if (Browser.DEBUG && window != mTab.mSubView) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800131 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
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400147 // 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
The Android Open Source Project0c908882009-03-03 19:32:16 -0800161 /**
162 * Private class for maintaining Tabs with a main WebView and a subwindow.
163 */
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400164 public class Tab implements WebView.PictureListener {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800165 // 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;
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400179 // Data used when displaying the tab in the picker.
180 private PickerData mPickerData;
181
The Android Open Source Project0c908882009-03-03 19:32:16 -0800182 // 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;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700191 // 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;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800197
Ben Murdochbff2d602009-07-01 20:19:05 +0100198 private ErrorConsoleView mErrorConsole;
199
The Android Open Source Project0c908882009-03-03 19:32:16 -0800200 // Construct a new tab
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700201 private Tab(WebView w, boolean closeOnExit, String appId, String url) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800202 mMainView = w;
203 mCloseOnExit = closeOnExit;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700204 mAppId = appId;
205 mOriginalUrl = url;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800206 }
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() {
Patrick Scott20abe042009-04-28 08:03:29 -0400254 if (mPickerData != null) {
255 return mPickerData.mUrl;
256 }
257 return null;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800258 }
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() {
Patrick Scott20abe042009-04-28 08:03:29 -0400268 if (mPickerData != null) {
269 return mPickerData.mTitle;
270 }
271 return null;
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400272 }
273
274 /**
275 * Returns the picker data.
276 */
277 public PickerData getPickerData() {
278 return mPickerData;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800279 }
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 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400340
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 }
The Android Open Source Project0c908882009-03-03 19:32:16 -0800352 };
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 /**
Ben Murdochbff2d602009-07-01 20:19:05 +0100393 * 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 /**
The Android Open Source Project0c908882009-03-03 19:32:16 -0800415 * 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) {
Patrick Scottae641ac2009-04-20 13:51:49 -0400473 if (tab == null) {
474 return -1;
475 }
The Android Open Source Project0c908882009-03-03 19:32:16 -0800476 return mTabs.indexOf(tab);
477 }
478
479 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700480 * Create a new tab.
The Android Open Source Project0c908882009-03-03 19:32:16 -0800481 * @return The newly createTab or null if we have reached the maximum
482 * number of open tabs.
483 */
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700484 Tab createNewTab(boolean closeOnExit, String appId, String url) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800485 int size = mTabs.size();
486 // Return false if we have maxed out on tabs
487 if (MAX_TABS == size) {
488 return null;
489 }
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700490 final WebView w = createNewWebView();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800491 // Create a new tab and add it to the tab list
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700492 Tab t = new Tab(w, closeOnExit, appId, url);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800493 mTabs.add(t);
The Android Open Source Project4e5f5872009-03-09 11:52:14 -0700494 // Initially put the tab in the background.
495 putTabInBackground(t);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800496 return t;
497 }
498
499 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700500 * 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 /**
The Android Open Source Project0c908882009-03-03 19:32:16 -0800508 * 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) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400553 if (t.mSavedState.containsKey(CURRPICTURE)) {
554 new File(t.mSavedState.getString(CURRPICTURE)).delete();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800555 }
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";
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400613 private static final String CURRWIDTH = "currentWidth";
614 private static final String CURRPICTURE = "currentPicture";
The Android Open Source Project0c908882009-03-03 19:32:16 -0800615 private static final String CLOSEONEXIT = "closeonexit";
616 private static final String PARENTTAB = "parentTab";
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700617 private static final String APPID = "appid";
618 private static final String ORIGINALURL = "originalUrl";
The Android Open Source Project0c908882009-03-03 19:32:16 -0800619
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) {
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700652 Tab t = createNewTab();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800653 // 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
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700664 Tab t = new Tab(null, false, null, null);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800665 t.mSavedState = inState.getBundle(WEBVIEW + i);
666 if (t.mSavedState != null) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400667 populatePickerDataFromSavedState(t);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700668 // 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);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800672 }
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() {
Grace Kloba92c18a52009-07-31 23:48:32 -0700701 if (getTabCount() == 0) return;
702
The Android Open Source Project0c908882009-03-03 19:32:16 -0800703 // free the least frequently used background tab
Grace Kloba92c18a52009-07-31 23:48:32 -0700704 Tab t = getLeastUsedTab(getCurrentTab().getParentTab());
The Android Open Source Project0c908882009-03-03 19:32:16 -0800705 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
Derek Sollenberger4433d032009-06-10 15:37:21 -0400713 // free the WebView's unused memory (this includes the cache)
714 Log.w(LOGTAG, "Free WebView's unused memory and cache");
The Android Open Source Project0c908882009-03-03 19:32:16 -0800715 WebView view = getCurrentWebView();
716 if (view != null) {
Derek Sollenberger4433d032009-06-10 15:37:21 -0400717 view.freeMemory();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800718 }
719 // force a gc
720 System.gc();
721 }
722
Grace Kloba92c18a52009-07-31 23:48:32 -0700723 private Tab getLeastUsedTab(Tab currentParent) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800724 // 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++);
Grace Kloba92c18a52009-07-31 23:48:32 -0700739 } while (i < queueSize
740 && ((t != null && t.mMainView == null) || t == currentParent));
The Android Open Source Project0c908882009-03-03 19:32:16 -0800741
Patrick Scottd068f802009-06-22 11:46:06 -0400742 // 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) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800745 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 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700817 * 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
Patrick Scottcd115892009-07-16 09:42:58 -0400834 // 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
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700873 /**
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;
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400906 t.mPickerData = null;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700907 // 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
The Android Open Source Projecta3c0aab2009-03-18 17:39:48 -0700919 // Enable the built-in zoom
920 w.getSettings().setBuiltInZoomControls(true);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700921 // 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 /**
The Android Open Source Project0c908882009-03-03 19:32:16 -0800929 * 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) {
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700934 return setCurrentTab(newTab, false);
935 }
936
Mike Reed7bfa63b2009-05-28 11:08:32 -0400937 /*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
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700979 /**
980 * If force is true, this method skips the check for newTab == current.
981 */
982 private boolean setCurrentTab(Tab newTab, boolean force) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800983 Tab current = getTab(mCurrentTab);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700984 if (current == newTab && !force) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800985 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;
The Android Open Source Project0c908882009-03-03 19:32:16 -08001004
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()
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001011 newTab.mMainView = mainView = createNewWebView();
The Android Open Source Project0c908882009-03-03 19:32:16 -08001012 }
Mike Reed7bfa63b2009-05-28 11:08:32 -04001013 putViewInForeground(mainView, mActivity.getWebViewClient(),
1014 mActivity.getWebChromeClient());
The Android Open Source Project0c908882009-03-03 19:32:16 -08001015 // Add the subwindow if it exists
1016 if (newTab.mSubViewContainer != null) {
Mike Reed7bfa63b2009-05-28 11:08:32 -04001017 putViewInForeground(newTab.mSubView, newTab.mSubViewClient,
1018 newTab.mSubViewChromeClient);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001019 }
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) {
Mike Reed7bfa63b2009-05-28 11:08:32 -04001033 putViewInBackground(t.mMainView);
1034 if (t.mSubView != null) {
1035 putViewInBackground(t.mSubView);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001036 }
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 /**
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001053 * Ensure that Tab t has data to display in the tab picker.
The Android Open Source Project0c908882009-03-03 19:32:16 -08001054 * @param t Tab to populate.
1055 */
1056 /* package */ void populatePickerData(Tab t) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001057 if (t == null) {
The Android Open Source Project0c908882009-03-03 19:32:16 -08001058 return;
1059 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001060
1061 // mMainView == null indicates that the tab has been freed.
1062 if (t.mMainView == null) {
1063 populatePickerDataFromSavedState(t);
1064 return;
1065 }
1066
The Android Open Source Project0c908882009-03-03 19:32:16 -08001067 // 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);
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001073
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();
The Android Open Source Project0c908882009-03-03 19:32:16 -08001082 }
1083
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001084 // 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.
The Android Open Source Project0c908882009-03-03 19:32:16 -08001110 }
1111 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001112
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();
Patrick Scottb0e4fc72009-07-14 10:49:22 -04001135
1136 // Remember the old picture if possible.
1137 if (t.mPickerData != null) {
1138 data.mPicture = t.mPickerData.mPicture;
1139 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001140 t.mPickerData = data;
The Android Open Source Project0c908882009-03-03 19:32:16 -08001141 }
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) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001151 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 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001159 }
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)) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001180 b.putString(CURRPICTURE, f.getPath());
The Android Open Source Project0c908882009-03-03 19:32:16 -08001181 }
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);
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001188
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);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001194 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001195 if (data.mTitle != null) {
1196 b.putString(CURRTITLE, data.mTitle);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001197 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001198 b.putInt(CURRWIDTH, data.mWidth);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001199 b.putBoolean(CLOSEONEXIT, t.mCloseOnExit);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001200 if (t.mAppId != null) {
1201 b.putString(APPID, t.mAppId);
1202 }
1203 if (t.mOriginalUrl != null) {
1204 b.putString(ORIGINALURL, t.mOriginalUrl);
1205 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001206
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 }
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001226 // 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;
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001229 t.mPickerData = null;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001230 t.mCloseOnExit = b.getBoolean(CLOSEONEXIT);
1231 t.mAppId = b.getString(APPID);
1232 t.mOriginalUrl = b.getString(ORIGINALURL);
1233
The Android Open Source Project0c908882009-03-03 19:32:16 -08001234 final WebView w = t.mMainView;
1235 final WebBackForwardList list = w.restoreState(b);
1236 if (list == null) {
1237 return false;
1238 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001239 if (b.containsKey(CURRPICTURE)) {
1240 final File f = new File(b.getString(CURRPICTURE));
The Android Open Source Project0c908882009-03-03 19:32:16 -08001241 w.restorePicture(b, f);
1242 f.delete();
1243 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001244 return true;
1245 }
1246}