blob: 1c8b7c83e2902172ef8cd9ab9ddbc49013480558 [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
198 // Construct a new tab
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700199 private Tab(WebView w, boolean closeOnExit, String appId, String url) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800200 mMainView = w;
201 mCloseOnExit = closeOnExit;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700202 mAppId = appId;
203 mOriginalUrl = url;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800204 }
205
206 /**
207 * Return the top window of this tab; either the subwindow if it is not
208 * null or the main window.
209 * @return The top window of this tab.
210 */
211 public WebView getTopWindow() {
212 if (mSubView != null) {
213 return mSubView;
214 }
215 return mMainView;
216 }
217
218 /**
219 * Return the main window of this tab. Note: if a tab is freed in the
220 * background, this can return null. It is only guaranteed to be
221 * non-null for the current tab.
222 * @return The main WebView of this tab.
223 */
224 public WebView getWebView() {
225 return mMainView;
226 }
227
228 /**
229 * Return the subwindow of this tab or null if there is no subwindow.
230 * @return The subwindow of this tab or null.
231 */
232 public WebView getSubWebView() {
233 return mSubView;
234 }
235
236 /**
237 * Return the subwindow container of this tab or null if there is no
238 * subwindow.
239 * @return The subwindow's container View.
240 */
241 public View getSubWebViewContainer() {
242 return mSubViewContainer;
243 }
244
245 /**
246 * Get the url of this tab. Valid after calling populatePickerData, but
247 * before calling wipePickerData, or if the webview has been destroyed.
248 *
249 * @return The WebView's url or null.
250 */
251 public String getUrl() {
Patrick Scott20abe042009-04-28 08:03:29 -0400252 if (mPickerData != null) {
253 return mPickerData.mUrl;
254 }
255 return null;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800256 }
257
258 /**
259 * Get the title of this tab. Valid after calling populatePickerData,
260 * but before calling wipePickerData, or if the webview has been
261 * destroyed. If the url has no title, use the url instead.
262 *
263 * @return The WebView's title (or url) or null.
264 */
265 public String getTitle() {
Patrick Scott20abe042009-04-28 08:03:29 -0400266 if (mPickerData != null) {
267 return mPickerData.mTitle;
268 }
269 return null;
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400270 }
271
272 /**
273 * Returns the picker data.
274 */
275 public PickerData getPickerData() {
276 return mPickerData;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800277 }
278
279 private void setParentTab(Tab parent) {
280 mParentTab = parent;
281 // This tab may have been freed due to low memory. If that is the
282 // case, the parent tab index is already saved. If we are changing
283 // that index (most likely due to removing the parent tab) we must
284 // update the parent tab index in the saved Bundle.
285 if (mSavedState != null) {
286 if (parent == null) {
287 mSavedState.remove(PARENTTAB);
288 } else {
289 mSavedState.putInt(PARENTTAB, getTabIndex(parent));
290 }
291 }
292 }
293
294 /**
295 * When a Tab is created through the content of another Tab, then
296 * we associate the Tabs.
297 * @param child the Tab that was created from this Tab
298 */
299 public void addChildTab(Tab child) {
300 if (mChildTabs == null) {
301 mChildTabs = new Vector<Tab>();
302 }
303 mChildTabs.add(child);
304 child.setParentTab(this);
305 }
306
307 private void removeFromTree() {
308 // detach the children
309 if (mChildTabs != null) {
310 for(Tab t : mChildTabs) {
311 t.setParentTab(null);
312 }
313 }
314
315 // Find myself in my parent list
316 if (mParentTab != null) {
317 mParentTab.mChildTabs.remove(this);
318 }
319 }
320
321 /**
322 * If this Tab was created through another Tab, then this method
323 * returns that Tab.
324 * @return the Tab parent or null
325 */
326 public Tab getParentTab() {
327 return mParentTab;
328 }
329
330 /**
331 * Return whether this tab should be closed when it is backing out of
332 * the first page.
333 * @return TRUE if this tab should be closed when exit.
334 */
335 public boolean closeOnExit() {
336 return mCloseOnExit;
337 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400338
339 public void onNewPicture(WebView view, Picture p) {
340 if (mPickerData == null) {
341 return;
342 }
343
344 mPickerData.mPicture = p;
345 // Tell the FakeWebView to redraw.
346 if (mPickerData.mFakeWebView != null) {
347 mPickerData.mFakeWebView.invalidate();
348 }
349 }
The Android Open Source Project0c908882009-03-03 19:32:16 -0800350 };
351
352 // Directory to store thumbnails for each WebView.
353 private final File mThumbnailDir;
354
355 /**
356 * Construct a new TabControl object that interfaces with the given
357 * BrowserActivity instance.
358 * @param activity A BrowserActivity instance that TabControl will interface
359 * with.
360 */
361 TabControl(BrowserActivity activity) {
362 mActivity = activity;
363 mInflateService =
364 ((LayoutInflater) activity.getSystemService(
365 Context.LAYOUT_INFLATER_SERVICE));
366 mThumbnailDir = activity.getDir("thumbnails", 0);
367 }
368
369 File getThumbnailDir() {
370 return mThumbnailDir;
371 }
372
373 BrowserActivity getBrowserActivity() {
374 return mActivity;
375 }
376
377 /**
378 * Return the current tab's main WebView. This will always return the main
379 * WebView for a given tab and not a subwindow.
380 * @return The current tab's WebView.
381 */
382 WebView getCurrentWebView() {
383 Tab t = getTab(mCurrentTab);
384 if (t == null) {
385 return null;
386 }
387 return t.mMainView;
388 }
389
390 /**
391 * Return the current tab's top-level WebView. This can return a subwindow
392 * if one exists.
393 * @return The top-level WebView of the current tab.
394 */
395 WebView getCurrentTopWebView() {
396 Tab t = getTab(mCurrentTab);
397 if (t == null) {
398 return null;
399 }
400 return t.mSubView != null ? t.mSubView : t.mMainView;
401 }
402
403 /**
404 * Return the current tab's subwindow if it exists.
405 * @return The subwindow of the current tab or null if it doesn't exist.
406 */
407 WebView getCurrentSubWindow() {
408 Tab t = getTab(mCurrentTab);
409 if (t == null) {
410 return null;
411 }
412 return t.mSubView;
413 }
414
415 /**
416 * Return the tab at the specified index.
417 * @return The Tab for the specified index or null if the tab does not
418 * exist.
419 */
420 Tab getTab(int index) {
421 if (index >= 0 && index < mTabs.size()) {
422 return mTabs.get(index);
423 }
424 return null;
425 }
426
427 /**
428 * Return the current tab.
429 * @return The current tab.
430 */
431 Tab getCurrentTab() {
432 return getTab(mCurrentTab);
433 }
434
435 /**
436 * Return the current tab index.
437 * @return The current tab index
438 */
439 int getCurrentIndex() {
440 return mCurrentTab;
441 }
442
443 /**
444 * Given a Tab, find it's index
445 * @param Tab to find
446 * @return index of Tab or -1 if not found
447 */
448 int getTabIndex(Tab tab) {
Patrick Scottae641ac2009-04-20 13:51:49 -0400449 if (tab == null) {
450 return -1;
451 }
The Android Open Source Project0c908882009-03-03 19:32:16 -0800452 return mTabs.indexOf(tab);
453 }
454
455 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700456 * Create a new tab.
The Android Open Source Project0c908882009-03-03 19:32:16 -0800457 * @return The newly createTab or null if we have reached the maximum
458 * number of open tabs.
459 */
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700460 Tab createNewTab(boolean closeOnExit, String appId, String url) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800461 int size = mTabs.size();
462 // Return false if we have maxed out on tabs
463 if (MAX_TABS == size) {
464 return null;
465 }
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700466 final WebView w = createNewWebView();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800467 // Create a new tab and add it to the tab list
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700468 Tab t = new Tab(w, closeOnExit, appId, url);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800469 mTabs.add(t);
The Android Open Source Project4e5f5872009-03-09 11:52:14 -0700470 // Initially put the tab in the background.
471 putTabInBackground(t);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800472 return t;
473 }
474
475 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700476 * Create a new tab with default values for closeOnExit(false),
477 * appId(null), and url(null).
478 */
479 Tab createNewTab() {
480 return createNewTab(false, null, null);
481 }
482
483 /**
The Android Open Source Project0c908882009-03-03 19:32:16 -0800484 * Remove the tab from the list. If the tab is the current tab shown, the
485 * last created tab will be shown.
486 * @param t The tab to be removed.
487 */
488 boolean removeTab(Tab t) {
489 if (t == null) {
490 return false;
491 }
492 // Only remove the tab if it is the current one.
493 if (getCurrentTab() == t) {
494 putTabInBackground(t);
495 }
496
497 // Only destroy the WebView if it still exists.
498 if (t.mMainView != null) {
499 // Take down the sub window.
500 dismissSubWindow(t);
501 // Remove the WebView's settings from the BrowserSettings list of
502 // observers.
503 BrowserSettings.getInstance().deleteObserver(
504 t.mMainView.getSettings());
505 // Destroy the main view and subview
506 t.mMainView.destroy();
507 t.mMainView = null;
508 }
509 // clear it's references to parent and children
510 t.removeFromTree();
511
512 // Remove it from our list of tabs.
513 mTabs.remove(t);
514
515 // The tab indices have shifted, update all the saved state so we point
516 // to the correct index.
517 for (Tab tab : mTabs) {
518 if (tab.mChildTabs != null) {
519 for (Tab child : tab.mChildTabs) {
520 child.setParentTab(tab);
521 }
522 }
523 }
524
525
526 // This tab may have been pushed in to the background and then closed.
527 // If the saved state contains a picture file, delete the file.
528 if (t.mSavedState != null) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400529 if (t.mSavedState.containsKey(CURRPICTURE)) {
530 new File(t.mSavedState.getString(CURRPICTURE)).delete();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800531 }
532 }
533
534 // Remove it from the queue of viewed tabs.
535 mTabQueue.remove(t);
536 mCurrentTab = -1;
537 return true;
538 }
539
540 /**
541 * Clear the back/forward list for all the current tabs.
542 */
543 void clearHistory() {
544 int size = getTabCount();
545 for (int i = 0; i < size; i++) {
546 Tab t = mTabs.get(i);
547 // TODO: if a tab is freed due to low memory, its history is not
548 // cleared here.
549 if (t.mMainView != null) {
550 t.mMainView.clearHistory();
551 }
552 if (t.mSubView != null) {
553 t.mSubView.clearHistory();
554 }
555 }
556 }
557
558 /**
559 * Destroy all the tabs and subwindows
560 */
561 void destroy() {
562 BrowserSettings s = BrowserSettings.getInstance();
563 for (Tab t : mTabs) {
564 if (t.mMainView != null) {
565 dismissSubWindow(t);
566 s.deleteObserver(t.mMainView.getSettings());
567 t.mMainView.destroy();
568 t.mMainView = null;
569 }
570 }
571 mTabs.clear();
572 mTabQueue.clear();
573 }
574
575 /**
576 * Returns the number of tabs created.
577 * @return The number of tabs created.
578 */
579 int getTabCount() {
580 return mTabs.size();
581 }
582
583 // Used for saving and restoring each Tab
584 private static final String WEBVIEW = "webview";
585 private static final String NUMTABS = "numTabs";
586 private static final String CURRTAB = "currentTab";
587 private static final String CURRURL = "currentUrl";
588 private static final String CURRTITLE = "currentTitle";
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400589 private static final String CURRWIDTH = "currentWidth";
590 private static final String CURRPICTURE = "currentPicture";
The Android Open Source Project0c908882009-03-03 19:32:16 -0800591 private static final String CLOSEONEXIT = "closeonexit";
592 private static final String PARENTTAB = "parentTab";
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700593 private static final String APPID = "appid";
594 private static final String ORIGINALURL = "originalUrl";
The Android Open Source Project0c908882009-03-03 19:32:16 -0800595
596 /**
597 * Save the state of all the Tabs.
598 * @param outState The Bundle to save the state to.
599 */
600 void saveState(Bundle outState) {
601 final int numTabs = getTabCount();
602 outState.putInt(NUMTABS, numTabs);
603 final int index = getCurrentIndex();
604 outState.putInt(CURRTAB, (index >= 0 && index < numTabs) ? index : 0);
605 for (int i = 0; i < numTabs; i++) {
606 final Tab t = getTab(i);
607 if (saveState(t)) {
608 outState.putBundle(WEBVIEW + i, t.mSavedState);
609 }
610 }
611 }
612
613 /**
614 * Restore the state of all the tabs.
615 * @param inState The saved state of all the tabs.
616 * @return True if there were previous tabs that were restored. False if
617 * there was no saved state or restoring the state failed.
618 */
619 boolean restoreState(Bundle inState) {
620 final int numTabs = (inState == null)
621 ? -1 : inState.getInt(NUMTABS, -1);
622 if (numTabs == -1) {
623 return false;
624 } else {
625 final int currentTab = inState.getInt(CURRTAB, -1);
626 for (int i = 0; i < numTabs; i++) {
627 if (i == currentTab) {
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700628 Tab t = createNewTab();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800629 // Me must set the current tab before restoring the state
630 // so that all the client classes are set.
631 setCurrentTab(t);
632 if (!restoreState(inState.getBundle(WEBVIEW + i), t)) {
633 Log.w(LOGTAG, "Fail in restoreState, load home page.");
634 t.mMainView.loadUrl(BrowserSettings.getInstance()
635 .getHomePage());
636 }
637 } else {
638 // Create a new tab and don't restore the state yet, add it
639 // to the tab list
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700640 Tab t = new Tab(null, false, null, null);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800641 t.mSavedState = inState.getBundle(WEBVIEW + i);
642 if (t.mSavedState != null) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400643 populatePickerDataFromSavedState(t);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700644 // Need to maintain the app id and original url so we
645 // can possibly reuse this tab.
646 t.mAppId = t.mSavedState.getString(APPID);
647 t.mOriginalUrl = t.mSavedState.getString(ORIGINALURL);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800648 }
649 mTabs.add(t);
650 mTabQueue.add(t);
651 }
652 }
653 // Rebuild the tree of tabs. Do this after all tabs have been
654 // created/restored so that the parent tab exists.
655 for (int i = 0; i < numTabs; i++) {
656 final Bundle b = inState.getBundle(WEBVIEW + i);
657 final Tab t = getTab(i);
658 if (b != null && t != null) {
659 final int parentIndex = b.getInt(PARENTTAB, -1);
660 if (parentIndex != -1) {
661 final Tab parent = getTab(parentIndex);
662 if (parent != null) {
663 parent.addChildTab(t);
664 }
665 }
666 }
667 }
668 }
669 return true;
670 }
671
672 /**
673 * Free the memory in this order, 1) free the background tab; 2) free the
674 * WebView cache;
675 */
676 void freeMemory() {
677 // free the least frequently used background tab
678 Tab t = getLeastUsedTab();
679 if (t != null) {
680 Log.w(LOGTAG, "Free a tab in the browser");
681 freeTab(t);
682 // force a gc
683 System.gc();
684 return;
685 }
686
Derek Sollenberger4433d032009-06-10 15:37:21 -0400687 // free the WebView's unused memory (this includes the cache)
688 Log.w(LOGTAG, "Free WebView's unused memory and cache");
The Android Open Source Project0c908882009-03-03 19:32:16 -0800689 WebView view = getCurrentWebView();
690 if (view != null) {
Derek Sollenberger4433d032009-06-10 15:37:21 -0400691 view.freeMemory();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800692 }
693 // force a gc
694 System.gc();
695 }
696
697 private Tab getLeastUsedTab() {
698 // Don't do anything if we only have 1 tab.
699 if (getTabCount() == 1) {
700 return null;
701 }
702
703 // Rip through the queue starting at the beginning and teardown the
704 // next available tab.
705 Tab t = null;
706 int i = 0;
707 final int queueSize = mTabQueue.size();
708 if (queueSize == 0) {
709 return null;
710 }
711 do {
712 t = mTabQueue.get(i++);
713 } while (i < queueSize && t != null && t.mMainView == null);
714
Patrick Scottd068f802009-06-22 11:46:06 -0400715 // Don't do anything if the last remaining tab is the current one or if
716 // the last tab has been freed already.
717 if (t == getCurrentTab() || t.mMainView == null) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800718 return null;
719 }
720
721 return t;
722 }
723
724 private void freeTab(Tab t) {
725 // Store the WebView's state.
726 saveState(t);
727
728 // Tear down the tab.
729 dismissSubWindow(t);
730 // Remove the WebView's settings from the BrowserSettings list of
731 // observers.
732 BrowserSettings.getInstance().deleteObserver(t.mMainView.getSettings());
733 t.mMainView.destroy();
734 t.mMainView = null;
735 }
736
737 /**
738 * Create a new subwindow unless a subwindow already exists.
739 * @return True if a new subwindow was created. False if one already exists.
740 */
741 void createSubWindow() {
742 Tab t = getTab(mCurrentTab);
743 if (t != null && t.mSubView == null) {
744 final View v = mInflateService.inflate(R.layout.browser_subwindow, null);
745 final WebView w = (WebView) v.findViewById(R.id.webview);
746 w.setMapTrackballToArrowKeys(false); // use trackball directly
747 final SubWindowClient subClient =
748 new SubWindowClient(mActivity.getWebViewClient());
749 final SubWindowChromeClient subChromeClient =
750 new SubWindowChromeClient(t,
751 mActivity.getWebChromeClient());
752 w.setWebViewClient(subClient);
753 w.setWebChromeClient(subChromeClient);
754 w.setDownloadListener(mActivity);
755 w.setOnCreateContextMenuListener(mActivity);
756 final BrowserSettings s = BrowserSettings.getInstance();
757 s.addObserver(w.getSettings()).update(s, null);
758 t.mSubView = w;
759 t.mSubViewClient = subClient;
760 t.mSubViewChromeClient = subChromeClient;
761 // FIXME: I really hate having to know the name of the view
762 // containing the webview.
763 t.mSubViewContainer = v.findViewById(R.id.subwindow_container);
764 final ImageButton cancel =
765 (ImageButton) v.findViewById(R.id.subwindow_close);
766 cancel.setOnClickListener(new OnClickListener() {
767 public void onClick(View v) {
768 subChromeClient.onCloseWindow(w);
769 }
770 });
771 }
772 }
773
774 /**
775 * Show the tab that contains the given WebView.
776 * @param view The WebView used to find the tab.
777 */
778 Tab getTabFromView(WebView view) {
779 final int size = getTabCount();
780 for (int i = 0; i < size; i++) {
781 final Tab t = getTab(i);
782 if (t.mSubView == view || t.mMainView == view) {
783 return t;
784 }
785 }
786 return null;
787 }
788
789 /**
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700790 * Return the tab with the matching application id.
791 * @param id The application identifier.
792 */
793 Tab getTabFromId(String id) {
794 if (id == null) {
795 return null;
796 }
797 final int size = getTabCount();
798 for (int i = 0; i < size; i++) {
799 final Tab t = getTab(i);
800 if (id.equals(t.mAppId)) {
801 return t;
802 }
803 }
804 return null;
805 }
806
Patrick Scottcd115892009-07-16 09:42:58 -0400807 // This method checks if a non-app tab (one created within the browser)
808 // matches the given url.
809 private boolean tabMatchesUrl(Tab t, String url) {
810 if (t.mAppId != null) {
811 return false;
812 } else if (t.mMainView == null) {
813 return false;
814 } else if (url.equals(t.mMainView.getUrl()) ||
815 url.equals(t.mMainView.getOriginalUrl())) {
816 return true;
817 }
818 return false;
819 }
820
821 /**
822 * Return the tab that has no app id associated with it and the url of the
823 * tab matches the given url.
824 * @param url The url to search for.
825 */
826 Tab findUnusedTabWithUrl(String url) {
827 if (url == null) {
828 return null;
829 }
830 // Check the current tab first.
831 Tab t = getCurrentTab();
832 if (t != null && tabMatchesUrl(t, url)) {
833 return t;
834 }
835 // Now check all the rest.
836 final int size = getTabCount();
837 for (int i = 0; i < size; i++) {
838 t = getTab(i);
839 if (tabMatchesUrl(t, url)) {
840 return t;
841 }
842 }
843 return null;
844 }
845
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700846 /**
847 * Recreate the main WebView of the given tab. Returns true if the WebView
848 * was deleted.
849 */
850 boolean recreateWebView(Tab t, String url) {
851 final WebView w = t.mMainView;
852 if (w != null) {
853 if (url != null && url.equals(t.mOriginalUrl)) {
854 // The original url matches the current url. Just go back to the
855 // first history item so we can load it faster than if we
856 // rebuilt the WebView.
857 final WebBackForwardList list = w.copyBackForwardList();
858 if (list != null) {
859 w.goBackOrForward(-list.getCurrentIndex());
860 w.clearHistory(); // maintains the current page.
861 return false;
862 }
863 }
864 // Remove the settings object from the global settings and destroy
865 // the WebView.
866 BrowserSettings.getInstance().deleteObserver(
867 t.mMainView.getSettings());
868 t.mMainView.destroy();
869 }
870 // Create a new WebView. If this tab is the current tab, we need to put
871 // back all the clients so force it to be the current tab.
872 t.mMainView = createNewWebView();
873 if (getCurrentTab() == t) {
874 setCurrentTab(t, true);
875 }
876 // Clear the saved state except for the app id and close-on-exit
877 // values.
878 t.mSavedState = null;
Patrick Scott2ed6edb2009-04-22 10:07:45 -0400879 t.mPickerData = null;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700880 // Save the new url in order to avoid deleting the WebView.
881 t.mOriginalUrl = url;
882 return true;
883 }
884
885 /**
886 * Creates a new WebView and registers it with the global settings.
887 */
888 private WebView createNewWebView() {
889 // Create a new WebView
890 WebView w = new WebView(mActivity);
891 w.setMapTrackballToArrowKeys(false); // use trackball directly
The Android Open Source Projecta3c0aab2009-03-18 17:39:48 -0700892 // Enable the built-in zoom
893 w.getSettings().setBuiltInZoomControls(true);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700894 // Add this WebView to the settings observer list and update the
895 // settings
896 final BrowserSettings s = BrowserSettings.getInstance();
897 s.addObserver(w.getSettings()).update(s, null);
898 return w;
899 }
900
901 /**
The Android Open Source Project0c908882009-03-03 19:32:16 -0800902 * Put the current tab in the background and set newTab as the current tab.
903 * @param newTab The new tab. If newTab is null, the current tab is not
904 * set.
905 */
906 boolean setCurrentTab(Tab newTab) {
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700907 return setCurrentTab(newTab, false);
908 }
909
Mike Reed7bfa63b2009-05-28 11:08:32 -0400910 /*package*/ void pauseCurrentTab() {
911 Tab t = getCurrentTab();
912 if (t != null) {
913 t.mMainView.onPause();
914 if (t.mSubView != null) {
915 t.mSubView.onPause();
916 }
917 }
918 }
919
920 /*package*/ void resumeCurrentTab() {
921 Tab t = getCurrentTab();
922 if (t != null) {
923 t.mMainView.onResume();
924 if (t.mSubView != null) {
925 t.mSubView.onResume();
926 }
927 }
928 }
929
930 private void putViewInForeground(WebView v, WebViewClient vc,
931 WebChromeClient cc) {
932 v.setWebViewClient(vc);
933 v.setWebChromeClient(cc);
934 v.setOnCreateContextMenuListener(mActivity);
935 v.setDownloadListener(mActivity);
936 v.onResume();
937 }
938
939 private void putViewInBackground(WebView v) {
940 // Set an empty callback so that default actions are not triggered.
941 v.setWebViewClient(mEmptyClient);
942 v.setWebChromeClient(mBackgroundChromeClient);
943 v.setOnCreateContextMenuListener(null);
944 // Leave the DownloadManager attached so that downloads can start in
945 // a non-active window. This can happen when going to a site that does
946 // a redirect after a period of time. The user could have switched to
947 // another tab while waiting for the download to start.
948 v.setDownloadListener(mActivity);
949 v.onPause();
950 }
951
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700952 /**
953 * If force is true, this method skips the check for newTab == current.
954 */
955 private boolean setCurrentTab(Tab newTab, boolean force) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800956 Tab current = getTab(mCurrentTab);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700957 if (current == newTab && !force) {
The Android Open Source Project0c908882009-03-03 19:32:16 -0800958 return true;
959 }
960 if (current != null) {
961 // Remove the current WebView and the container of the subwindow
962 putTabInBackground(current);
963 }
964
965 if (newTab == null) {
966 return false;
967 }
968
969 // Move the newTab to the end of the queue
970 int index = mTabQueue.indexOf(newTab);
971 if (index != -1) {
972 mTabQueue.remove(index);
973 }
974 mTabQueue.add(newTab);
975
976 WebView mainView;
The Android Open Source Project0c908882009-03-03 19:32:16 -0800977
978 // Display the new current tab
979 mCurrentTab = mTabs.indexOf(newTab);
980 mainView = newTab.mMainView;
981 boolean needRestore = (mainView == null);
982 if (needRestore) {
983 // Same work as in createNewTab() except don't do new Tab()
The Android Open Source Projectf59ec872009-03-13 13:04:24 -0700984 newTab.mMainView = mainView = createNewWebView();
The Android Open Source Project0c908882009-03-03 19:32:16 -0800985 }
Mike Reed7bfa63b2009-05-28 11:08:32 -0400986 putViewInForeground(mainView, mActivity.getWebViewClient(),
987 mActivity.getWebChromeClient());
The Android Open Source Project0c908882009-03-03 19:32:16 -0800988 // Add the subwindow if it exists
989 if (newTab.mSubViewContainer != null) {
Mike Reed7bfa63b2009-05-28 11:08:32 -0400990 putViewInForeground(newTab.mSubView, newTab.mSubViewClient,
991 newTab.mSubViewChromeClient);
The Android Open Source Project0c908882009-03-03 19:32:16 -0800992 }
993 if (needRestore) {
994 // Have to finish setCurrentTab work before calling restoreState
995 if (!restoreState(newTab.mSavedState, newTab)) {
996 mainView.loadUrl(BrowserSettings.getInstance().getHomePage());
997 }
998 }
999 return true;
1000 }
1001
1002 /*
1003 * Put the tab in the background using all the empty/background clients.
1004 */
1005 private void putTabInBackground(Tab t) {
Mike Reed7bfa63b2009-05-28 11:08:32 -04001006 putViewInBackground(t.mMainView);
1007 if (t.mSubView != null) {
1008 putViewInBackground(t.mSubView);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001009 }
1010 }
1011
1012 /*
1013 * Dismiss the subwindow for the given tab.
1014 */
1015 void dismissSubWindow(Tab t) {
1016 if (t != null && t.mSubView != null) {
1017 BrowserSettings.getInstance().deleteObserver(
1018 t.mSubView.getSettings());
1019 t.mSubView.destroy();
1020 t.mSubView = null;
1021 t.mSubViewContainer = null;
1022 }
1023 }
1024
1025 /**
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001026 * Ensure that Tab t has data to display in the tab picker.
The Android Open Source Project0c908882009-03-03 19:32:16 -08001027 * @param t Tab to populate.
1028 */
1029 /* package */ void populatePickerData(Tab t) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001030 if (t == null) {
The Android Open Source Project0c908882009-03-03 19:32:16 -08001031 return;
1032 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001033
1034 // mMainView == null indicates that the tab has been freed.
1035 if (t.mMainView == null) {
1036 populatePickerDataFromSavedState(t);
1037 return;
1038 }
1039
The Android Open Source Project0c908882009-03-03 19:32:16 -08001040 // FIXME: The only place we cared about subwindow was for
1041 // bookmarking (i.e. not when saving state). Was this deliberate?
1042 final WebBackForwardList list = t.mMainView.copyBackForwardList();
1043 final WebHistoryItem item =
1044 list != null ? list.getCurrentItem() : null;
1045 populatePickerData(t, item);
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001046
1047 // This method is only called during the tab picker creation. At this
1048 // point we need to listen for new pictures since the WebView is still
1049 // active.
1050 final WebView w = t.getTopWindow();
1051 w.setPictureListener(t);
1052 // Capture the picture here instead of populatePickerData since it can
1053 // be called when saving the state of a tab.
1054 t.mPickerData.mPicture = w.capturePicture();
The Android Open Source Project0c908882009-03-03 19:32:16 -08001055 }
1056
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001057 // Create the PickerData and populate it using the saved state of the tab.
1058 private void populatePickerDataFromSavedState(Tab t) {
1059 if (t.mSavedState == null) {
1060 return;
1061 }
1062
1063 final PickerData data = new PickerData();
1064 final Bundle state = t.mSavedState;
1065 data.mUrl = state.getString(CURRURL);
1066 data.mTitle = state.getString(CURRTITLE);
1067 data.mWidth = state.getInt(CURRWIDTH, 0);
1068 // XXX: These keys are from WebView.savePicture so if they change, this
1069 // will break.
1070 data.mScale = state.getFloat("scale", 1.0f);
1071 data.mScrollX = state.getInt("scrollX", 0);
1072 data.mScrollY = state.getInt("scrollY", 0);
1073
1074 if (state.containsKey(CURRPICTURE)) {
1075 final File f = new File(t.mSavedState.getString(CURRPICTURE));
1076 try {
1077 final FileInputStream in = new FileInputStream(f);
1078 data.mPicture = Picture.createFromStream(in);
1079 in.close();
1080 } catch (Exception ex) {
1081 // Ignore any problems with inflating the picture. We just
1082 // won't draw anything.
The Android Open Source Project0c908882009-03-03 19:32:16 -08001083 }
1084 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001085
1086 // Set the tab's picker data.
1087 t.mPickerData = data;
1088 }
1089
1090 // Populate the picker data using the given history item and the current
1091 // top WebView.
1092 private void populatePickerData(Tab t, WebHistoryItem item) {
1093 final PickerData data = new PickerData();
1094 if (item != null) {
1095 data.mUrl = item.getUrl();
1096 data.mTitle = item.getTitle();
1097 if (data.mTitle == null) {
1098 data.mTitle = data.mUrl;
1099 }
1100 }
1101 // We want to display the top window in the tab picker but use the url
1102 // and title of the main window.
1103 final WebView w = t.getTopWindow();
1104 data.mWidth = w.getWidth();
1105 data.mScale = w.getScale();
1106 data.mScrollX = w.getScrollX();
1107 data.mScrollY = w.getScrollY();
Patrick Scottb0e4fc72009-07-14 10:49:22 -04001108
1109 // Remember the old picture if possible.
1110 if (t.mPickerData != null) {
1111 data.mPicture = t.mPickerData.mPicture;
1112 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001113 t.mPickerData = data;
The Android Open Source Project0c908882009-03-03 19:32:16 -08001114 }
1115
1116 /**
1117 * Clean up the data for all tabs.
1118 */
1119 /* package */ void wipeAllPickerData() {
1120 int size = getTabCount();
1121 for (int i = 0; i < size; i++) {
1122 final Tab t = getTab(i);
1123 if (t != null && t.mSavedState == null) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001124 t.mPickerData = null;
1125 }
1126 if (t.mMainView != null) {
1127 // Clear the picture listeners.
1128 t.mMainView.setPictureListener(null);
1129 if (t.mSubView != null) {
1130 t.mSubView.setPictureListener(null);
1131 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001132 }
1133 }
1134 }
1135
1136 /*
1137 * Save the state for an individual tab.
1138 */
1139 private boolean saveState(Tab t) {
1140 if (t != null) {
1141 final WebView w = t.mMainView;
1142 // If the WebView is null it means we ran low on memory and we
1143 // already stored the saved state in mSavedState.
1144 if (w == null) {
1145 return true;
1146 }
1147 final Bundle b = new Bundle();
1148 final WebBackForwardList list = w.saveState(b);
1149 if (list != null) {
1150 final File f = new File(mThumbnailDir, w.hashCode()
1151 + "_pic.save");
1152 if (w.savePicture(b, f)) {
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001153 b.putString(CURRPICTURE, f.getPath());
The Android Open Source Project0c908882009-03-03 19:32:16 -08001154 }
1155 }
1156
1157 // Store some extra info for displaying the tab in the picker.
1158 final WebHistoryItem item =
1159 list != null ? list.getCurrentItem() : null;
1160 populatePickerData(t, item);
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001161
1162 // XXX: WebView.savePicture stores the scale and scroll positions
1163 // in the bundle so we don't have to do it here.
1164 final PickerData data = t.mPickerData;
1165 if (data.mUrl != null) {
1166 b.putString(CURRURL, data.mUrl);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001167 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001168 if (data.mTitle != null) {
1169 b.putString(CURRTITLE, data.mTitle);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001170 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001171 b.putInt(CURRWIDTH, data.mWidth);
The Android Open Source Project0c908882009-03-03 19:32:16 -08001172 b.putBoolean(CLOSEONEXIT, t.mCloseOnExit);
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001173 if (t.mAppId != null) {
1174 b.putString(APPID, t.mAppId);
1175 }
1176 if (t.mOriginalUrl != null) {
1177 b.putString(ORIGINALURL, t.mOriginalUrl);
1178 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001179
1180 // Remember the parent tab so the relationship can be restored.
1181 if (t.mParentTab != null) {
1182 b.putInt(PARENTTAB, getTabIndex(t.mParentTab));
1183 }
1184
1185 // Remember the saved state.
1186 t.mSavedState = b;
1187 return true;
1188 }
1189 return false;
1190 }
1191
1192 /*
1193 * Restore the state of the tab.
1194 */
1195 private boolean restoreState(Bundle b, Tab t) {
1196 if (b == null) {
1197 return false;
1198 }
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001199 // Restore the internal state even if the WebView fails to restore.
1200 // This will maintain the app id, original url and close-on-exit values.
1201 t.mSavedState = null;
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001202 t.mPickerData = null;
The Android Open Source Projectf59ec872009-03-13 13:04:24 -07001203 t.mCloseOnExit = b.getBoolean(CLOSEONEXIT);
1204 t.mAppId = b.getString(APPID);
1205 t.mOriginalUrl = b.getString(ORIGINALURL);
1206
The Android Open Source Project0c908882009-03-03 19:32:16 -08001207 final WebView w = t.mMainView;
1208 final WebBackForwardList list = w.restoreState(b);
1209 if (list == null) {
1210 return false;
1211 }
Patrick Scott2ed6edb2009-04-22 10:07:45 -04001212 if (b.containsKey(CURRPICTURE)) {
1213 final File f = new File(b.getString(CURRPICTURE));
The Android Open Source Project0c908882009-03-03 19:32:16 -08001214 w.restorePicture(b, f);
1215 f.delete();
1216 }
The Android Open Source Project0c908882009-03-03 19:32:16 -08001217 return true;
1218 }
1219}