summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--res/values-sw600dp/dimensions.xml2
-rw-r--r--res/values/dimensions.xml3
-rw-r--r--src/com/android/browser/BaseUi.java14
-rw-r--r--src/com/android/browser/BrowserActivity.java4
-rw-r--r--src/com/android/browser/Controller.java59
-rw-r--r--src/com/android/browser/CrashRecoveryHandler.java2
-rw-r--r--src/com/android/browser/PhoneUi.java5
-rw-r--r--src/com/android/browser/PieControlPhone.java2
-rw-r--r--src/com/android/browser/PieControlXLarge.java2
-rw-r--r--src/com/android/browser/PreloadController.java5
-rw-r--r--src/com/android/browser/SnapshotTab.java18
-rw-r--r--src/com/android/browser/Tab.java273
-rw-r--r--src/com/android/browser/TabControl.java43
-rw-r--r--src/com/android/browser/UI.java2
-rw-r--r--src/com/android/browser/WebViewController.java2
-rw-r--r--src/com/android/browser/XLargeUi.java9
-rw-r--r--src/com/android/browser/provider/BrowserProvider2.java60
17 files changed, 363 insertions, 142 deletions
diff --git a/res/values-sw600dp/dimensions.xml b/res/values-sw600dp/dimensions.xml
index 2c4aaae..bc87333 100644
--- a/res/values-sw600dp/dimensions.xml
+++ b/res/values-sw600dp/dimensions.xml
@@ -37,4 +37,6 @@
<dimen name="suggest_item_padding">16dp</dimen>
<dimen name="toolbar_height">48dip</dimen>
<dimen name="progress_bar_margin">-11dip</dimen>
+ <dimen name="tab_thumbnail_width">@dimen/qc_thumb_width</dimen>
+ <dimen name="tab_thumbnail_height">@dimen/qc_thumb_height</dimen>
</resources>
diff --git a/res/values/dimensions.xml b/res/values/dimensions.xml
index 3b57f19..8fca68a 100644
--- a/res/values/dimensions.xml
+++ b/res/values/dimensions.xml
@@ -67,7 +67,8 @@
<dimen name="menu_width">240dip</dimen>
<dimen name="menu_height">32dip</dimen>
<dimen name="toolbar_height">52dip</dimen>
- <dimen name="tab_capture_size">160dp</dimen>
+ <dimen name="tab_thumbnail_width">@dimen/nav_tab_width</dimen>
+ <dimen name="tab_thumbnail_height">@dimen/nav_tab_height</dimen>
<dimen name="nav_tab_width">240dip</dimen>
<dimen name="nav_tab_height">160dip</dimen>
<dimen name="nav_tab_text_normal">18sp</dimen>
diff --git a/src/com/android/browser/BaseUi.java b/src/com/android/browser/BaseUi.java
index b270dea..091126c 100644
--- a/src/com/android/browser/BaseUi.java
+++ b/src/com/android/browser/BaseUi.java
@@ -738,20 +738,6 @@ public abstract class BaseUi implements UI, OnTouchListener {
warning.show();
}
- protected void captureTab(final Tab tab) {
- captureTab(tab,
- (int) mActivity.getResources()
- .getDimension(R.dimen.qc_thumb_width),
- (int) mActivity.getResources()
- .getDimension(R.dimen.qc_thumb_height));
- }
-
- protected void captureTab(final Tab tab, int width, int height) {
- if ((tab == null) || (tab.getWebView() == null)) return;
- Bitmap sshot = Controller.createScreenshot(tab, width, height);
- tab.setScreenshot(sshot);
- }
-
protected WebView getWebView() {
if (mActiveTab != null) {
return mActiveTab.getWebView();
diff --git a/src/com/android/browser/BrowserActivity.java b/src/com/android/browser/BrowserActivity.java
index 0df2e94..6ec6071 100644
--- a/src/com/android/browser/BrowserActivity.java
+++ b/src/com/android/browser/BrowserActivity.java
@@ -112,7 +112,7 @@ public class BrowserActivity extends Activity {
protected void onNewIntent(Intent intent) {
if (ACTION_RESTART.equals(intent.getAction())) {
Bundle outState = new Bundle();
- mController.onSaveInstanceState(outState, true);
+ mController.onSaveInstanceState(outState);
finish();
getApplicationContext().startActivity(
new Intent(getApplicationContext(), BrowserActivity.class)
@@ -163,7 +163,7 @@ public class BrowserActivity extends Activity {
if (LOGV_ENABLED) {
Log.v(LOGTAG, "BrowserActivity.onSaveInstanceState: this=" + this);
}
- mController.onSaveInstanceState(outState, true);
+ mController.onSaveInstanceState(outState);
}
@Override
diff --git a/src/com/android/browser/Controller.java b/src/com/android/browser/Controller.java
index b3be618..92f448c 100644
--- a/src/com/android/browser/Controller.java
+++ b/src/com/android/browser/Controller.java
@@ -80,6 +80,7 @@ import com.android.browser.IntentHandler.UrlData;
import com.android.browser.UI.ComboViews;
import com.android.browser.UI.DropdownChangeListener;
import com.android.browser.provider.BrowserProvider;
+import com.android.browser.provider.BrowserProvider2.Thumbnails;
import com.android.browser.provider.SnapshotProvider.Snapshots;
import com.android.browser.search.SearchEngine;
import com.android.common.Search;
@@ -87,6 +88,7 @@ import com.android.common.Search;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.net.URLEncoder;
+import java.util.ArrayList;
import java.util.Calendar;
import java.util.HashMap;
import java.util.List;
@@ -309,6 +311,7 @@ public class Controller
private void onPreloginFinished(Bundle icicle, Intent intent, long currentTabId,
boolean restoreIncognitoTabs) {
if (currentTabId == -1) {
+ BackgroundHandler.execute(new PruneThumbnails(mActivity, null));
final Bundle extra = intent.getExtras();
// Create an initial tab.
// If the intent is ACTION_VIEW and data is not null, the Browser is
@@ -335,7 +338,13 @@ public class Controller
} else {
mTabControl.restoreState(icicle, currentTabId, restoreIncognitoTabs,
mUi.needsRestoreAllTabs());
- mUi.updateTabs(mTabControl.getTabs());
+ List<Tab> tabs = mTabControl.getTabs();
+ ArrayList<Long> restoredTabs = new ArrayList<Long>(tabs.size());
+ for (Tab t : tabs) {
+ restoredTabs.add(t.getId());
+ }
+ BackgroundHandler.execute(new PruneThumbnails(mActivity, restoredTabs));
+ mUi.updateTabs(tabs);
// TabControl.restoreState() will create a new tab even if
// restoring the state fails.
setActiveTab(mTabControl.getCurrentTab());
@@ -357,6 +366,38 @@ public class Controller
}
}
+ private static class PruneThumbnails implements Runnable {
+ private Context mContext;
+ private List<Long> mIds;
+
+ PruneThumbnails(Context context, List<Long> preserveIds) {
+ mContext = context.getApplicationContext();
+ mIds = preserveIds;
+ }
+
+ @Override
+ public void run() {
+ ContentResolver cr = mContext.getContentResolver();
+ if (mIds == null || mIds.size() == 0) {
+ cr.delete(Thumbnails.CONTENT_URI, null, null);
+ } else {
+ int length = mIds.size();
+ StringBuilder where = new StringBuilder();
+ where.append(Thumbnails._ID);
+ where.append(" not in (");
+ for (int i = 0; i < length; i++) {
+ where.append(mIds.get(i));
+ if (i < (length - 1)) {
+ where.append(",");
+ }
+ }
+ where.append(")");
+ cr.delete(Thumbnails.CONTENT_URI, where.toString(), null);
+ }
+ }
+
+ }
+
@Override
public WebViewFactory getWebViewFactory() {
return mFactory;
@@ -612,7 +653,7 @@ public class Controller
}
- void onSaveInstanceState(Bundle outState, boolean saveImages) {
+ void onSaveInstanceState(Bundle outState) {
// the default implementation requires each view to have an id. As the
// browser handles the state itself and it doesn't use id for the views,
// don't call the default implementation. Otherwise it will trigger the
@@ -620,7 +661,7 @@ public class Controller
// focused view XXX has no id".
// Save all the tabs
- mTabControl.saveState(outState, false);
+ mTabControl.saveState(outState);
if (!outState.isEmpty()) {
// Save time so that we know how old incognito tabs (if any) are.
outState.putSerializable("lastActiveDate", Calendar.getInstance());
@@ -1902,13 +1943,6 @@ public class Controller
R.dimen.bookmarkThumbnailHeight);
}
- static Bitmap createScreenshot(Tab tab, int width, int height) {
- if ((tab != null) && (tab.getWebView() != null)) {
- return createScreenshot(tab.getWebView(), width, height);
- }
- return null;
- }
-
static Bitmap createScreenshot(WebView view, int width, int height) {
// We render to a bitmap 2x the desired size so that we can then
// re-scale it with filtering since canvas.scale doesn't filter
@@ -2646,4 +2680,9 @@ public class Controller
return true;
}
+ @Override
+ public boolean shouldCaptureThumbnails() {
+ return mUi.shouldCaptureThumbnails();
+ }
+
}
diff --git a/src/com/android/browser/CrashRecoveryHandler.java b/src/com/android/browser/CrashRecoveryHandler.java
index 02636c0..fdcdbc6 100644
--- a/src/com/android/browser/CrashRecoveryHandler.java
+++ b/src/com/android/browser/CrashRecoveryHandler.java
@@ -133,7 +133,7 @@ public class CrashRecoveryHandler {
public void run() {
try {
final Bundle state = new Bundle();
- mController.onSaveInstanceState(state, false);
+ mController.onSaveInstanceState(state);
Message.obtain(mBackgroundHandler, MSG_WRITE_STATE, state)
.sendToTarget();
// Remove any queued up saves
diff --git a/src/com/android/browser/PhoneUi.java b/src/com/android/browser/PhoneUi.java
index 28db6f0..1c9d5a0 100644
--- a/src/com/android/browser/PhoneUi.java
+++ b/src/com/android/browser/PhoneUi.java
@@ -276,4 +276,9 @@ public class PhoneUi extends BaseUi {
}
}
+ @Override
+ public boolean shouldCaptureThumbnails() {
+ return true;
+ }
+
}
diff --git a/src/com/android/browser/PieControlPhone.java b/src/com/android/browser/PieControlPhone.java
index c4b28fa..0b62cef 100644
--- a/src/com/android/browser/PieControlPhone.java
+++ b/src/com/android/browser/PieControlPhone.java
@@ -72,7 +72,7 @@ public class PieControlPhone extends PieControlBase implements OnClickListener {
private void buildTabs() {
final List<Tab> tabs = mUiController.getTabs();
- mUi.captureTab(mUi.getActiveTab());
+ mUi.getActiveTab().capture();
mTabAdapter.setTabs(tabs);
PieStackView sym = (PieStackView) mShowTabs.getPieView();
sym.setCurrent(mUiController.getTabControl().getCurrentPosition());
diff --git a/src/com/android/browser/PieControlXLarge.java b/src/com/android/browser/PieControlXLarge.java
index a036e0d..95f586e 100644
--- a/src/com/android/browser/PieControlXLarge.java
+++ b/src/com/android/browser/PieControlXLarge.java
@@ -112,7 +112,7 @@ public class PieControlXLarge extends PieControlBase implements OnClickListener
private void buildTabs() {
final List<Tab> tabs = mUiController.getTabs();
- mUi.captureTab(mUi.getActiveTab());
+ mUi.getActiveTab().capture();
mTabAdapter.setTabs(tabs);
PieStackView sym = (PieStackView) mShowTabs.getPieView();
sym.setCurrent(mUiController.getTabControl().getCurrentPosition());
diff --git a/src/com/android/browser/PreloadController.java b/src/com/android/browser/PreloadController.java
index 652ea8e..08e223f 100644
--- a/src/com/android/browser/PreloadController.java
+++ b/src/com/android/browser/PreloadController.java
@@ -276,4 +276,9 @@ public class PreloadController implements WebViewController {
if (LOGD_ENABLED) Log.d(LOGTAG, "hideAutoLogin()");
}
+ @Override
+ public boolean shouldCaptureThumbnails() {
+ return false;
+ }
+
}
diff --git a/src/com/android/browser/SnapshotTab.java b/src/com/android/browser/SnapshotTab.java
index e57502f..bd6dd5b 100644
--- a/src/com/android/browser/SnapshotTab.java
+++ b/src/com/android/browser/SnapshotTab.java
@@ -22,6 +22,7 @@ import android.database.Cursor;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.AsyncTask;
+import android.os.Bundle;
import android.util.Log;
import android.webkit.WebView;
@@ -44,7 +45,7 @@ public class SnapshotTab extends Tab {
private boolean mIsLive;
public SnapshotTab(WebViewController wvcontroller, long snapshotId) {
- super(wvcontroller, null);
+ super(wvcontroller, null, null);
mSnapshotId = snapshotId;
mWebViewFactory = mWebViewController.getWebViewFactory();
WebView web = mWebViewFactory.createWebView(false);
@@ -98,8 +99,8 @@ public class SnapshotTab extends Tab {
}
@Override
- boolean saveState() {
- return false;
+ Bundle saveState() {
+ return null;
}
public long getDateCreated() {
@@ -198,4 +199,15 @@ public class SnapshotTab extends Tab {
}
}
+
+ @Override
+ protected void persistThumbnail() {
+ // Nope
+ }
+
+ @Override
+ protected void deleteThumbnail() {
+ // Nope
+ }
+
}
diff --git a/src/com/android/browser/Tab.java b/src/com/android/browser/Tab.java
index beac2ff..a4bcc99 100644
--- a/src/com/android/browser/Tab.java
+++ b/src/com/android/browser/Tab.java
@@ -20,11 +20,13 @@ import android.app.Activity;
import android.app.AlertDialog;
import android.app.SearchManager;
import android.content.ContentResolver;
+import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnCancelListener;
import android.content.Intent;
+import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.BitmapFactory;
@@ -53,7 +55,6 @@ import android.webkit.HttpAuthHandler;
import android.webkit.SslErrorHandler;
import android.webkit.URLUtil;
import android.webkit.ValueCallback;
-import android.webkit.WebBackForwardList;
import android.webkit.WebBackForwardListClient;
import android.webkit.WebChromeClient;
import android.webkit.WebHistoryItem;
@@ -68,10 +69,12 @@ import android.widget.TextView;
import android.widget.Toast;
import com.android.browser.homepages.HomeProvider;
+import com.android.browser.provider.BrowserProvider2.Thumbnails;
import com.android.browser.provider.SnapshotProvider.Snapshots;
import com.android.common.speech.LoggingEvents;
import java.io.ByteArrayOutputStream;
+import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
@@ -95,6 +98,8 @@ class Tab implements PictureListener {
private static final int MSG_CAPTURE = 42;
private static final int CAPTURE_DELAY = 500;
+ private static Bitmap sDefaultFavicon;
+
public enum LockIcon {
LOCK_ICON_UNSECURE,
LOCK_ICON_SECURE,
@@ -161,6 +166,13 @@ class Tab implements PictureListener {
private Bitmap mCapture;
private Handler mHandler;
+ private static synchronized Bitmap getDefaultFavicon(Context context) {
+ if (sDefaultFavicon == null) {
+ sDefaultFavicon = BitmapFactory.decodeResource(
+ context.getResources(), R.drawable.app_web_browser_sm);
+ }
+ return sDefaultFavicon;
+ }
// All the state needed for a page
protected static class PageState {
@@ -179,8 +191,7 @@ class Tab implements PictureListener {
mOriginalUrl = mUrl = "";
mTitle = c.getString(R.string.new_tab);
}
- mFavicon = BitmapFactory.decodeResource(
- c.getResources(), R.drawable.app_web_browser_sm);
+ mFavicon = null;
mLockIcon = LockIcon.LOCK_ICON_UNSECURE;
}
@@ -192,13 +203,9 @@ class Tab implements PictureListener {
} else {
mLockIcon = LockIcon.LOCK_ICON_UNSECURE;
}
- if (favicon != null) {
- mFavicon = favicon;
- } else {
- mFavicon = BitmapFactory.decodeResource(
- c.getResources(), R.drawable.app_web_browser_sm);
- }
+ mFavicon = favicon;
}
+
}
// The current/loading page's state
@@ -211,7 +218,6 @@ class Tab implements PictureListener {
static final String PARENTTAB = "parentTab";
static final String APPID = "appid";
static final String INCOGNITO = "privateBrowsingEnabled";
- static final String SCREENSHOT = "screenshot";
static final String USERAGENT = "useragent";
// -------------------------------------------------------------------------
@@ -576,19 +582,7 @@ class Tab implements PictureListener {
url, SystemClock.uptimeMillis() - mLoadStartTime);
}
mInPageLoad = false;
- // Sync state (in case of stop/timeout)
- mCurrentState.mUrl = view.getUrl();
- if (mCurrentState.mUrl == null) {
- mCurrentState.mUrl = url != null ? url : "";
- }
- mCurrentState.mOriginalUrl = view.getOriginalUrl();
- mCurrentState.mTitle = view.getTitle();
- mCurrentState.mFavicon = view.getFavicon();
- if (!URLUtil.isHttpsUrl(mCurrentState.mUrl)) {
- // In case we stop when loading an HTTPS page from an HTTP page
- // but before a provisional load occurred
- mCurrentState.mLockIcon = LockIcon.LOCK_ICON_UNSECURE;
- }
+ syncCurrentState(view, url);
mWebViewController.onPageFinished(Tab.this);
}
@@ -894,6 +888,22 @@ class Tab implements PictureListener {
};
+ private void syncCurrentState(WebView view, String url) {
+ // Sync state (in case of stop/timeout)
+ mCurrentState.mUrl = view.getUrl();
+ if (mCurrentState.mUrl == null) {
+ mCurrentState.mUrl = url != null ? url : "";
+ }
+ mCurrentState.mOriginalUrl = view.getOriginalUrl();
+ mCurrentState.mTitle = view.getTitle();
+ mCurrentState.mFavicon = view.getFavicon();
+ if (!URLUtil.isHttpsUrl(mCurrentState.mUrl)) {
+ // In case we stop when loading an HTTPS page from an HTTP page
+ // but before a provisional load occurred
+ mCurrentState.mLockIcon = LockIcon.LOCK_ICON_UNSECURE;
+ }
+ }
+
// Called by DeviceAccountLogin when the Tab needs to have the auto-login UI
// displayed.
void setDeviceAccountLogin(DeviceAccountLogin login) {
@@ -1355,11 +1365,16 @@ class Tab implements PictureListener {
// -------------------------------------------------------------------------
- // TODO temporarily use activity here
- // remove later
-
// Construct a new tab
Tab(WebViewController wvcontroller, WebView w) {
+ this(wvcontroller, w, null);
+ }
+
+ Tab(WebViewController wvcontroller, Bundle state) {
+ this(wvcontroller, null, state);
+ }
+
+ Tab(WebViewController wvcontroller, WebView w, Bundle state) {
mWebViewController = wvcontroller;
mContext = mWebViewController.getContext();
mSettings = BrowserSettings.getInstance();
@@ -1393,21 +1408,46 @@ class Tab implements PictureListener {
}
};
+ mCaptureWidth = mContext.getResources().getDimensionPixelSize(
+ R.dimen.tab_thumbnail_width);
+ mCaptureHeight = mContext.getResources().getDimensionPixelSize(
+ R.dimen.tab_thumbnail_height);
+ updateShouldCaptureThumbnails();
+ restoreState(state);
setWebView(w);
- mCaptureWidth = mContext.getResources().getDimensionPixelSize(R.dimen.nav_tab_width);
- mCaptureHeight = mContext.getResources().getDimensionPixelSize(R.dimen.nav_tab_height);
- mCapture = Bitmap.createBitmap(mCaptureWidth, mCaptureHeight,
- Bitmap.Config.RGB_565);
mHandler = new Handler() {
public void handleMessage(Message m) {
- Tab.this.capture();
+ switch (m.what) {
+ case MSG_CAPTURE:
+ capture();
+ break;
+ }
}
};
+ }
+ public void updateShouldCaptureThumbnails() {
+ if (mWebViewController.shouldCaptureThumbnails()) {
+ synchronized (Tab.this) {
+ if (mCapture == null) {
+ mCapture = Bitmap.createBitmap(mCaptureWidth, mCaptureHeight,
+ Bitmap.Config.RGB_565);
+ if (mInForeground) {
+ postCapture();
+ }
+ }
+ }
+ } else {
+ synchronized (Tab.this) {
+ mCapture = null;
+ deleteThumbnail();
+ }
+ }
}
public void setController(WebViewController ctl) {
mWebViewController = ctl;
+ updateShouldCaptureThumbnails();
}
public void setId(long id) {
@@ -1435,6 +1475,13 @@ class Tab implements PictureListener {
mWebViewController.onSetWebView(this, w);
+ if (mMainView != null) {
+ if (w != null) {
+ syncCurrentState(w, null);
+ } else {
+ mCurrentState = new PageState(mContext, false);
+ }
+ }
// set the new one
mMainView = w;
// attach the WebViewClient, WebChromeClient and DownloadListener
@@ -1448,6 +1495,10 @@ class Tab implements PictureListener {
mMainView.setDownloadListener(mDownloadListener);
mMainView.setWebBackForwardListClient(mWebBackForwardListClient);
mMainView.setPictureListener(this);
+ if (mSavedState != null) {
+ mMainView.restoreState(mSavedState);
+ mSavedState = null;
+ }
}
}
@@ -1480,6 +1531,7 @@ class Tab implements PictureListener {
if (mParent != null) {
mParent.mChildren.remove(this);
}
+ deleteThumbnail();
}
/**
@@ -1739,7 +1791,10 @@ class Tab implements PictureListener {
* Get the favicon of this tab.
*/
Bitmap getFavicon() {
- return mCurrentState.mFavicon;
+ if (mCurrentState.mFavicon != null) {
+ return mCurrentState.mFavicon;
+ }
+ return getDefaultFavicon(mContext);
}
public boolean isBookmarkedSite() {
@@ -1796,43 +1851,19 @@ class Tab implements PictureListener {
}
/**
- * Get the cached saved state bundle.
- * @return cached state bundle
+ * @return The Bundle with the tab's state if it can be saved, otherwise null
*/
- Bundle getSavedState() {
- return mSavedState;
- }
-
- Bundle getSavedState(boolean saveImages) {
- if (saveImages && mCapture != null) {
- Bundle b = new Bundle(mSavedState);
- b.putParcelable(SCREENSHOT, mCapture);
- return b;
- }
- return mSavedState;
- }
-
- /**
- * Set the saved state.
- */
- void setSavedState(Bundle state) {
- mSavedState = state;
- }
-
- /**
- * @return TRUE if succeed in saving the state.
- */
- boolean saveState() {
+ public Bundle saveState() {
// If the WebView is null it means we ran low on memory and we already
// stored the saved state in mSavedState.
if (mMainView == null) {
- return mSavedState != null;
+ return mSavedState;
}
// If the tab is the homepage or has no URL, don't save it
String homepage = BrowserSettings.getInstance().getHomePage();
if (TextUtils.equals(homepage, mCurrentState.mUrl)
|| TextUtils.isEmpty(mCurrentState.mUrl)) {
- return false;
+ return null;
}
mSavedState = new Bundle();
@@ -1841,6 +1872,7 @@ class Tab implements PictureListener {
mSavedState.putLong(ID, mId);
mSavedState.putString(CURRURL, mCurrentState.mUrl);
mSavedState.putString(CURRTITLE, mCurrentState.mTitle);
+ mSavedState.putBoolean(INCOGNITO, mMainView.isPrivateBrowsingEnabled());
if (mAppId != null) {
mSavedState.putString(APPID, mAppId);
}
@@ -1850,35 +1882,35 @@ class Tab implements PictureListener {
}
mSavedState.putBoolean(USERAGENT,
mSettings.hasDesktopUseragent(getWebView()));
- return true;
+ return mSavedState;
}
/*
* Restore the state of the tab.
*/
- boolean restoreState(Bundle b) {
- if (b == null) {
- return false;
+ private void restoreState(Bundle b) {
+ mSavedState = b;
+ if (mSavedState == null) {
+ return;
}
// Restore the internal state even if the WebView fails to restore.
// This will maintain the app id, original url and close-on-exit values.
- mSavedState = null;
mId = b.getLong(ID);
mAppId = b.getString(APPID);
- final Bitmap sshot = b.getParcelable(SCREENSHOT);
- if (sshot != null) {
- mCapture = sshot;
- }
if (b.getBoolean(USERAGENT)
!= mSettings.hasDesktopUseragent(getWebView())) {
mSettings.toggleDesktopUseragent(getWebView());
}
-
- final WebBackForwardList list = mMainView.restoreState(b);
- if (list == null) {
- return false;
+ String url = b.getString(CURRURL);
+ String title = b.getString(CURRTITLE);
+ boolean incognito = b.getBoolean(INCOGNITO);
+ mCurrentState = new PageState(mContext, incognito, url, null);
+ mCurrentState.mTitle = title;
+ synchronized (Tab.this) {
+ if (mCapture != null) {
+ BackgroundHandler.execute(mLoadThumbnail);
+ }
}
- return true;
}
public void updateBookmarkedStatus() {
@@ -1896,12 +1928,10 @@ class Tab implements PictureListener {
}
};
- public void setScreenshot(Bitmap screenshot) {
- mCapture = screenshot;
- }
-
public Bitmap getScreenshot() {
- return mCapture;
+ synchronized (Tab.this) {
+ return mCapture;
+ }
}
public boolean isSnapshot() {
@@ -1963,11 +1993,16 @@ class Tab implements PictureListener {
float scale = mCaptureWidth / (float) mMainView.getWidth();
c.scale(scale, scale, left, top);
mMainView.draw(c);
+ persistThumbnail();
}
@Override
public void onNewPicture(WebView view, Picture picture) {
//update screenshot
+ postCapture();
+ }
+
+ private void postCapture() {
if (!mHandler.hasMessages(MSG_CAPTURE)) {
mHandler.sendEmptyMessageDelayed(MSG_CAPTURE, CAPTURE_DELAY);
}
@@ -1993,4 +2028,84 @@ class Tab implements PictureListener {
}
}
+ protected void persistThumbnail() {
+ BackgroundHandler.execute(mSaveThumbnail);
+ }
+
+ protected void deleteThumbnail() {
+ BackgroundHandler.execute(mDeleteThumbnail);
+ }
+
+ private void updateCaptureFromBlob(byte[] blob) {
+ synchronized (Tab.this) {
+ if (mCapture == null) {
+ return;
+ }
+ mCapture.copyPixelsFromBuffer(ByteBuffer.wrap(blob));
+ }
+ }
+
+ private byte[] getCaptureBlob() {
+ synchronized (Tab.this) {
+ if (mCapture == null) {
+ return null;
+ }
+ ByteBuffer buffer = ByteBuffer.allocate(mCapture.getByteCount());
+ mCapture.copyPixelsToBuffer(buffer);
+ return buffer.array();
+ }
+ }
+
+ private Runnable mSaveThumbnail = new Runnable() {
+
+ @Override
+ public void run() {
+ byte[] blob = getCaptureBlob();
+ if (blob == null) {
+ return;
+ }
+ ContentResolver cr = mContext.getContentResolver();
+ ContentValues values = new ContentValues();
+ values.put(Thumbnails._ID, mId);
+ values.put(Thumbnails.THUMBNAIL, blob);
+ cr.insert(Thumbnails.CONTENT_URI, values);
+ }
+ };
+
+ private Runnable mDeleteThumbnail = new Runnable() {
+
+ @Override
+ public void run() {
+ ContentResolver cr = mContext.getContentResolver();
+ try {
+ cr.delete(ContentUris.withAppendedId(Thumbnails.CONTENT_URI, mId),
+ null, null);
+ } catch (Throwable t) {}
+ }
+ };
+
+ private Runnable mLoadThumbnail = new Runnable() {
+
+ @Override
+ public void run() {
+ ContentResolver cr = mContext.getContentResolver();
+ Cursor c = null;
+ try {
+ Uri uri = ContentUris.withAppendedId(Thumbnails.CONTENT_URI, mId);
+ c = cr.query(uri, new String[] {Thumbnails._ID,
+ Thumbnails.THUMBNAIL}, null, null, null);
+ if (c.moveToFirst()) {
+ byte[] data = c.getBlob(1);
+ if (data != null && data.length > 0) {
+ updateCaptureFromBlob(data);
+ }
+ }
+ } finally {
+ if (c != null) {
+ c.close();
+ }
+ }
+ }
+ };
+
}
diff --git a/src/com/android/browser/TabControl.java b/src/com/android/browser/TabControl.java
index cd8da2e..b708841 100644
--- a/src/com/android/browser/TabControl.java
+++ b/src/com/android/browser/TabControl.java
@@ -186,6 +186,12 @@ class TabControl {
* number of open tabs.
*/
Tab createNewTab(boolean privateBrowsing) {
+ return createNewTab(null, privateBrowsing);
+ }
+
+ Tab createNewTab(Bundle state, boolean privateBrowsing) {
+ int size = mTabs.size();
+ // Return false if we have maxed out on tabs
if (!canCreateNewTab()) {
return null;
}
@@ -193,7 +199,7 @@ class TabControl {
final WebView w = createNewWebView(privateBrowsing);
// Create a new tab and add it to the tab list
- Tab t = new Tab(mController, w);
+ Tab t = new Tab(mController, w, state);
t.setId(getNextId());
mTabs.add(t);
// Initially put the tab in the background.
@@ -288,7 +294,7 @@ class TabControl {
* @param outState
* @param saveImages
*/
- void saveState(Bundle outState, boolean saveImages) {
+ void saveState(Bundle outState) {
final int numTabs = getTabCount();
if (numTabs == 0) {
return;
@@ -296,10 +302,10 @@ class TabControl {
long[] ids = new long[numTabs];
int i = 0;
for (Tab tab : mTabs) {
- if (tab.saveState()) {
+ Bundle tabState = tab.saveState();
+ if (tabState != null) {
ids[i++] = tab.getId();
- outState.putBundle(Long.toString(tab.getId()),
- tab.getSavedState(saveImages));
+ outState.putBundle(Long.toString(tab.getId()), tabState);
} else {
ids[i++] = -1;
}
@@ -329,7 +335,7 @@ class TabControl {
final long oldcurrent = inState.getLong(CURRENT);
long current = -1;
if (restoreIncognitoTabs || (hasState(oldcurrent, inState) && !isIncognito(oldcurrent, inState))) {
- current = oldcurrent;
+ current = oldcurrent;
} else {
// pick first non incognito tab
for (long id : ids) {
@@ -363,8 +369,6 @@ class TabControl {
* @param restoreIncognitoTabs Restoring private browsing tabs
* @param restoreAll All webviews get restored, not just the current tab
* (this does not override handling of incognito tabs)
- * @return True if there were previous tabs that were restored. False if
- * there was no saved state or restoring the state failed.
*/
void restoreState(Bundle inState, long currentId,
boolean restoreIncognitoTabs, boolean restoreAll) {
@@ -387,7 +391,7 @@ class TabControl {
&& state.getBoolean(Tab.INCOGNITO)) {
// ignore tab
} else if (id == currentId || restoreAll) {
- Tab t = createNewTab();
+ Tab t = createNewTab(state, false);
if (t == null) {
// We could "break" at this point, but we want
// sNextId to be set correctly.
@@ -399,23 +403,12 @@ class TabControl {
if (id == currentId) {
setCurrentTab(t);
}
- if (!t.restoreState(state)) {
- Log.w(LOGTAG, "Fail in restoreState, load home page.");
- t.getWebView().loadUrl(BrowserSettings.getInstance()
- .getHomePage());
- }
} else {
// Create a new tab and don't restore the state yet, add it
// to the tab list
- Tab t = new Tab(mController, null);
+ Tab t = new Tab(mController, state);
t.setId(id);
tabMap.put(id, t);
- if (state != null) {
- t.setSavedState(state);
- // Need to maintain the app id and original url so we
- // can possibly reuse this tab.
- t.setAppId(state.getString(Tab.APPID));
- }
mTabs.add(t);
// added the tab to the front as they are not current
mTabQueue.add(0, t);
@@ -619,8 +612,6 @@ class TabControl {
if (getCurrentTab() == t) {
setCurrentTab(t, true);
}
- // Clear the saved state and picker data
- t.setSavedState(null);
}
/**
@@ -681,12 +672,6 @@ class TabControl {
newTab.setWebView(mainView);
}
newTab.putInForeground();
- if (needRestore) {
- // Have to finish setCurrentTab work before calling restoreState
- if (!newTab.restoreState(newTab.getSavedState())) {
- mainView.loadUrl(BrowserSettings.getInstance().getHomePage());
- }
- }
return true;
}
diff --git a/src/com/android/browser/UI.java b/src/com/android/browser/UI.java
index 23897f7..2984d5c 100644
--- a/src/com/android/browser/UI.java
+++ b/src/com/android/browser/UI.java
@@ -147,4 +147,6 @@ public interface UI {
void setUseQuickControls(boolean enabled);
+ public boolean shouldCaptureThumbnails();
+
}
diff --git a/src/com/android/browser/WebViewController.java b/src/com/android/browser/WebViewController.java
index 175cbf8..f4ff764 100644
--- a/src/com/android/browser/WebViewController.java
+++ b/src/com/android/browser/WebViewController.java
@@ -122,4 +122,6 @@ public interface WebViewController {
void showAutoLogin(Tab tab);
void hideAutoLogin(Tab tab);
+
+ boolean shouldCaptureThumbnails();
}
diff --git a/src/com/android/browser/XLargeUi.java b/src/com/android/browser/XLargeUi.java
index 73821bf..aeba7cf 100644
--- a/src/com/android/browser/XLargeUi.java
+++ b/src/com/android/browser/XLargeUi.java
@@ -101,6 +101,10 @@ public class XLargeUi extends BaseUi {
setTitleGravity(Gravity.NO_GRAVITY);
}
mTabBar.setUseQuickControls(mUseQuickControls);
+ // We need to update the tabs with this change
+ for (Tab t : mTabControl.getTabs()) {
+ t.updateShouldCaptureThumbnails();
+ }
}
private void checkTabCount() {
@@ -334,4 +338,9 @@ public class XLargeUi extends BaseUi {
return mTabBar;
}
+ @Override
+ public boolean shouldCaptureThumbnails() {
+ return mUseQuickControls;
+ }
+
}
diff --git a/src/com/android/browser/provider/BrowserProvider2.java b/src/com/android/browser/provider/BrowserProvider2.java
index b974c0e..e40a882 100644
--- a/src/com/android/browser/provider/BrowserProvider2.java
+++ b/src/com/android/browser/provider/BrowserProvider2.java
@@ -74,6 +74,13 @@ public class BrowserProvider2 extends SQLiteContentProvider {
static final Uri LEGACY_AUTHORITY_URI = new Uri.Builder()
.authority(LEGACY_AUTHORITY).scheme("content").build();
+ public static interface Thumbnails {
+ public static final Uri CONTENT_URI = Uri.withAppendedPath(
+ BrowserContract.AUTHORITY_URI, "thumbnails");
+ public static final String _ID = "_id";
+ public static final String THUMBNAIL = "thumbnail";
+ }
+
static final String TABLE_BOOKMARKS = "bookmarks";
static final String TABLE_HISTORY = "history";
static final String TABLE_IMAGES = "images";
@@ -81,6 +88,7 @@ public class BrowserProvider2 extends SQLiteContentProvider {
static final String TABLE_SYNC_STATE = "syncstate";
static final String TABLE_SETTINGS = "settings";
static final String TABLE_SNAPSHOTS = "snapshots";
+ static final String TABLE_THUMBNAILS = "thumbnails";
static final String TABLE_BOOKMARKS_JOIN_IMAGES = "bookmarks LEFT OUTER JOIN images " +
"ON bookmarks.url = images." + Images.URL;
@@ -111,6 +119,9 @@ public class BrowserProvider2 extends SQLiteContentProvider {
"WHERE url IS NOT NULL AND deleted == 0) AND url_key NOT IN " +
"(SELECT url FROM history WHERE url IS NOT NULL)";
+ static final int THUMBNAILS = 10;
+ static final int THUMBNAILS_ID = 11;
+
static final int BOOKMARKS = 1000;
static final int BOOKMARKS_ID = 1001;
static final int BOOKMARKS_FOLDER = 1002;
@@ -187,6 +198,8 @@ public class BrowserProvider2 extends SQLiteContentProvider {
matcher.addURI(authority, "combined", COMBINED);
matcher.addURI(authority, "combined/#", COMBINED_ID);
matcher.addURI(authority, "settings", SETTINGS);
+ matcher.addURI(authority, "thumbnails", THUMBNAILS);
+ matcher.addURI(authority, "thumbnails/#", THUMBNAILS_ID);
// Legacy
matcher.addURI(LEGACY_AUTHORITY, "searches", SEARCHES);
@@ -333,7 +346,7 @@ public class BrowserProvider2 extends SQLiteContentProvider {
final class DatabaseHelper extends SQLiteOpenHelper {
static final String DATABASE_NAME = "browser2.db";
- static final int DATABASE_VERSION = 30;
+ static final int DATABASE_VERSION = 31;
public DatabaseHelper(Context context) {
super(context, DATABASE_NAME, null, DATABASE_VERSION);
}
@@ -396,6 +409,7 @@ public class BrowserProvider2 extends SQLiteContentProvider {
");");
createAccountsView(db);
+ createThumbnails(db);
mSyncHelper.createDatabase(db);
@@ -406,6 +420,13 @@ public class BrowserProvider2 extends SQLiteContentProvider {
enableSync(db);
}
+ void createThumbnails(SQLiteDatabase db) {
+ db.execSQL("CREATE TABLE IF NOT EXISTS " + TABLE_THUMBNAILS + " (" +
+ Thumbnails._ID + " INTEGER PRIMARY KEY," +
+ Thumbnails.THUMBNAIL + " BLOB NOT NULL" +
+ ");");
+ }
+
void enableSync(SQLiteDatabase db) {
ContentValues values = new ContentValues();
values.put(Settings.KEY, Settings.KEY_SYNC_ENABLED);
@@ -500,6 +521,9 @@ public class BrowserProvider2 extends SQLiteContentProvider {
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
+ if (oldVersion < 31) {
+ createThumbnails(db);
+ }
if (oldVersion < 30) {
db.execSQL("DROP VIEW IF EXISTS " + VIEW_SNAPSHOTS_COMBINED);
db.execSQL("DROP TABLE IF EXISTS " + TABLE_SNAPSHOTS);
@@ -974,6 +998,18 @@ public class BrowserProvider2 extends SQLiteContentProvider {
break;
}
+ case THUMBNAILS_ID: {
+ selection = DatabaseUtils.concatenateWhere(
+ selection, Thumbnails._ID + " = ?");
+ selectionArgs = DatabaseUtils.appendSelectionArgs(selectionArgs,
+ new String[] { Long.toString(ContentUris.parseId(uri)) });
+ // fall through
+ }
+ case THUMBNAILS: {
+ qb.setTables(TABLE_THUMBNAILS);
+ break;
+ }
+
default: {
throw new UnsupportedOperationException("Unknown URL " + uri.toString());
}
@@ -1173,6 +1209,17 @@ public class BrowserProvider2 extends SQLiteContentProvider {
}
break;
}
+ case THUMBNAILS_ID: {
+ selection = DatabaseUtils.concatenateWhere(
+ selection, Thumbnails._ID + " = ?");
+ selectionArgs = DatabaseUtils.appendSelectionArgs(selectionArgs,
+ new String[] { Long.toString(ContentUris.parseId(uri)) });
+ // fall through
+ }
+ case THUMBNAILS: {
+ deleted = db.delete(TABLE_THUMBNAILS, selection, selectionArgs);
+ break;
+ }
default: {
throw new UnsupportedOperationException("Unknown delete URI " + uri);
}
@@ -1310,6 +1357,11 @@ public class BrowserProvider2 extends SQLiteContentProvider {
break;
}
+ case THUMBNAILS: {
+ id = db.replaceOrThrow(TABLE_THUMBNAILS, null, values);
+ break;
+ }
+
default: {
throw new UnsupportedOperationException("Unknown insert URI " + uri);
}
@@ -1552,6 +1604,12 @@ public class BrowserProvider2 extends SQLiteContentProvider {
break;
}
+ case THUMBNAILS: {
+ modified = db.update(TABLE_THUMBNAILS, values,
+ selection, selectionArgs);
+ break;
+ }
+
default: {
throw new UnsupportedOperationException("Unknown update URI " + uri);
}