blob: 909c2a3dbeb053a10d7ad739d526d636bc14e557 [file] [log] [blame]
/*
* Copyright (C) 2010 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.browser;
import android.content.ContentResolver;
import android.content.ContentUris;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteException;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Handler;
import android.os.Message;
import android.provider.BrowserContract;
import android.provider.BrowserContract.History;
import android.util.Log;
import com.android.browser.provider.BrowserProvider2.Thumbnails;
import java.nio.ByteBuffer;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
public class DataController {
private static final String LOGTAG = "DataController";
// Message IDs
private static final int HISTORY_UPDATE_VISITED = 100;
private static final int HISTORY_UPDATE_TITLE = 101;
private static final int QUERY_URL_IS_BOOKMARK = 200;
private static final int TAB_LOAD_THUMBNAIL = 201;
private static final int TAB_SAVE_THUMBNAIL = 202;
private static final int TAB_DELETE_THUMBNAIL = 203;
private static DataController sInstance;
private Context mContext;
private DataControllerHandler mDataHandler;
private Handler mCbHandler; // To respond on the UI thread
private ByteBuffer mBuffer; // to capture thumbnails
/* package */ static interface OnQueryUrlIsBookmark {
void onQueryUrlIsBookmark(String url, boolean isBookmark);
}
private static class CallbackContainer {
Object replyTo;
Object[] args;
}
private static class DCMessage {
int what;
Object obj;
Object replyTo;
DCMessage(int w, Object o) {
what = w;
obj = o;
}
}
/* package */ static DataController getInstance(Context c) {
if (sInstance == null) {
sInstance = new DataController(c);
}
return sInstance;
}
private DataController(Context c) {
mContext = c.getApplicationContext();
mDataHandler = new DataControllerHandler();
mDataHandler.start();
mCbHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
CallbackContainer cc = (CallbackContainer) msg.obj;
switch (msg.what) {
case QUERY_URL_IS_BOOKMARK: {
OnQueryUrlIsBookmark cb = (OnQueryUrlIsBookmark) cc.replyTo;
String url = (String) cc.args[0];
boolean isBookmark = (Boolean) cc.args[1];
cb.onQueryUrlIsBookmark(url, isBookmark);
break;
}
}
}
};
}
public void updateVisitedHistory(String url) {
mDataHandler.sendMessage(HISTORY_UPDATE_VISITED, url);
}
public void updateHistoryTitle(String url, String title) {
mDataHandler.sendMessage(HISTORY_UPDATE_TITLE, new String[] { url, title });
}
public void queryBookmarkStatus(String url, OnQueryUrlIsBookmark replyTo) {
if (url == null || url.trim().length() == 0) {
// null or empty url is never a bookmark
replyTo.onQueryUrlIsBookmark(url, false);
return;
}
mDataHandler.sendMessage(QUERY_URL_IS_BOOKMARK, url.trim(), replyTo);
}
public void loadThumbnail(Tab tab) {
mDataHandler.sendMessage(TAB_LOAD_THUMBNAIL, tab);
}
public void deleteThumbnail(Tab tab) {
mDataHandler.sendMessage(TAB_DELETE_THUMBNAIL, tab.getId());
}
public void saveThumbnail(Tab tab) {
mDataHandler.sendMessage(TAB_SAVE_THUMBNAIL, tab);
}
// The standard Handler and Message classes don't allow the queue manipulation
// we want (such as peeking). So we use our own queue.
class DataControllerHandler extends Thread {
private BlockingQueue<DCMessage> mMessageQueue
= new LinkedBlockingQueue<DCMessage>();
public DataControllerHandler() {
super("DataControllerHandler");
}
@Override
public void run() {
setPriority(Thread.MIN_PRIORITY);
while (true) {
try {
handleMessage(mMessageQueue.take());
} catch (InterruptedException ex) {
break;
}
}
}
void sendMessage(int what, Object obj) {
DCMessage m = new DCMessage(what, obj);
mMessageQueue.add(m);
}
void sendMessage(int what, Object obj, Object replyTo) {
DCMessage m = new DCMessage(what, obj);
m.replyTo = replyTo;
mMessageQueue.add(m);
}
private void handleMessage(DCMessage msg) {
switch (msg.what) {
case HISTORY_UPDATE_VISITED:
doUpdateVisitedHistory((String) msg.obj);
break;
case HISTORY_UPDATE_TITLE:
String[] args = (String[]) msg.obj;
doUpdateHistoryTitle(args[0], args[1]);
break;
case QUERY_URL_IS_BOOKMARK:
// TODO: Look for identical messages in the queue and remove them
// TODO: Also, look for partial matches and merge them (such as
// multiple callbacks querying the same URL)
doQueryBookmarkStatus((String) msg.obj, msg.replyTo);
break;
case TAB_LOAD_THUMBNAIL:
doLoadThumbnail((Tab) msg.obj);
break;
case TAB_DELETE_THUMBNAIL:
ContentResolver cr = mContext.getContentResolver();
try {
cr.delete(ContentUris.withAppendedId(
Thumbnails.CONTENT_URI, (Long)msg.obj),
null, null);
} catch (Throwable t) {}
break;
case TAB_SAVE_THUMBNAIL:
doSaveThumbnail((Tab)msg.obj);
break;
}
}
private byte[] getCaptureBlob(Tab tab) {
synchronized (tab) {
Bitmap capture = tab.getScreenshot();
if (capture == null) {
return null;
}
if (mBuffer == null || mBuffer.limit() < capture.getByteCount()) {
mBuffer = ByteBuffer.allocate(capture.getByteCount());
}
capture.copyPixelsToBuffer(mBuffer);
mBuffer.rewind();
return mBuffer.array();
}
}
private void doSaveThumbnail(Tab tab) {
byte[] blob = getCaptureBlob(tab);
if (blob == null) {
return;
}
ContentResolver cr = mContext.getContentResolver();
ContentValues values = new ContentValues();
values.put(Thumbnails._ID, tab.getId());
values.put(Thumbnails.THUMBNAIL, blob);
cr.insert(Thumbnails.CONTENT_URI, values);
}
private void doLoadThumbnail(Tab tab) {
ContentResolver cr = mContext.getContentResolver();
Cursor c = null;
try {
Uri uri = ContentUris.withAppendedId(Thumbnails.CONTENT_URI, tab.getId());
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) {
tab.updateCaptureFromBlob(data);
}
}
} finally {
if (c != null) {
c.close();
}
}
}
private void doUpdateVisitedHistory(String url) {
ContentResolver cr = mContext.getContentResolver();
Cursor c = null;
try {
c = cr.query(History.CONTENT_URI, new String[] { History._ID, History.VISITS },
History.URL + "=?", new String[] { url }, null);
if (c.moveToFirst()) {
ContentValues values = new ContentValues();
values.put(History.VISITS, c.getInt(1) + 1);
values.put(History.DATE_LAST_VISITED, System.currentTimeMillis());
cr.update(ContentUris.withAppendedId(History.CONTENT_URI, c.getLong(0)),
values, null, null);
} else {
android.provider.Browser.truncateHistory(cr);
ContentValues values = new ContentValues();
values.put(History.URL, url);
values.put(History.VISITS, 1);
values.put(History.DATE_LAST_VISITED, System.currentTimeMillis());
values.put(History.TITLE, url);
values.put(History.DATE_CREATED, 0);
values.put(History.USER_ENTERED, 0);
cr.insert(History.CONTENT_URI, values);
}
} finally {
if (c != null) c.close();
}
}
private void doQueryBookmarkStatus(String url, Object replyTo) {
// Check to see if the site is bookmarked
Cursor cursor = null;
boolean isBookmark = false;
try {
cursor = mContext.getContentResolver().query(
BookmarkUtils.getBookmarksUri(mContext),
new String[] { BrowserContract.Bookmarks.URL },
BrowserContract.Bookmarks.URL + " == ?",
new String[] { url },
null);
isBookmark = cursor.moveToFirst();
} catch (SQLiteException e) {
Log.e(LOGTAG, "Error checking for bookmark: " + e);
} finally {
if (cursor != null) cursor.close();
}
CallbackContainer cc = new CallbackContainer();
cc.replyTo = replyTo;
cc.args = new Object[] { url, isBookmark };
mCbHandler.obtainMessage(QUERY_URL_IS_BOOKMARK, cc).sendToTarget();
}
private void doUpdateHistoryTitle(String url, String title) {
ContentResolver cr = mContext.getContentResolver();
ContentValues values = new ContentValues();
values.put(History.TITLE, title);
cr.update(History.CONTENT_URI, values, History.URL + "=?",
new String[] { url });
}
}
}