mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-12-11 16:32:59 +00:00
Bug 917394 - Rearrange about:home tabs as per new design (r=sriram)
--HG-- rename : mobile/android/base/home/MostVisitedPage.java => mobile/android/base/home/TopSitesPage.java rename : mobile/android/base/resources/layout/home_most_visited_page.xml => mobile/android/base/resources/layout/home_top_sites_page.xml
This commit is contained in:
parent
356f594ec7
commit
ed9279760c
@ -1401,7 +1401,7 @@ abstract public class BrowserApp extends GeckoApp
|
|||||||
animator.setUseHardwareLayer(false);
|
animator.setUseHardwareLayer(false);
|
||||||
|
|
||||||
mBrowserToolbar.startEditing(url, animator);
|
mBrowserToolbar.startEditing(url, animator);
|
||||||
showHomePagerWithAnimator(HomePager.Page.HISTORY, animator);
|
showHomePagerWithAnimator(HomePager.Page.TOP_SITES, animator);
|
||||||
|
|
||||||
animator.start();
|
animator.start();
|
||||||
}
|
}
|
||||||
|
@ -230,7 +230,6 @@ FENNEC_JAVA_FILES = \
|
|||||||
home/FaviconsLoader.java \
|
home/FaviconsLoader.java \
|
||||||
home/LastTabsPage.java \
|
home/LastTabsPage.java \
|
||||||
home/MostRecentPage.java \
|
home/MostRecentPage.java \
|
||||||
home/MostVisitedPage.java \
|
|
||||||
home/MultiTypeCursorAdapter.java \
|
home/MultiTypeCursorAdapter.java \
|
||||||
home/PinBookmarkDialog.java \
|
home/PinBookmarkDialog.java \
|
||||||
home/ReadingListPage.java \
|
home/ReadingListPage.java \
|
||||||
@ -243,6 +242,7 @@ FENNEC_JAVA_FILES = \
|
|||||||
home/TopBookmarkItemView.java \
|
home/TopBookmarkItemView.java \
|
||||||
home/TopBookmarksAdapter.java \
|
home/TopBookmarksAdapter.java \
|
||||||
home/TopBookmarksView.java \
|
home/TopBookmarksView.java \
|
||||||
|
home/TopSitesPage.java \
|
||||||
home/TwoLinePageRow.java \
|
home/TwoLinePageRow.java \
|
||||||
menu/GeckoMenu.java \
|
menu/GeckoMenu.java \
|
||||||
menu/GeckoMenuInflater.java \
|
menu/GeckoMenuInflater.java \
|
||||||
@ -478,12 +478,12 @@ RES_LAYOUT = \
|
|||||||
res/layout/home_last_tabs_page.xml \
|
res/layout/home_last_tabs_page.xml \
|
||||||
res/layout/home_history_list.xml \
|
res/layout/home_history_list.xml \
|
||||||
res/layout/home_most_recent_page.xml \
|
res/layout/home_most_recent_page.xml \
|
||||||
res/layout/home_most_visited_page.xml \
|
|
||||||
res/layout/home_pager.xml \
|
res/layout/home_pager.xml \
|
||||||
res/layout/home_reading_list_page.xml \
|
res/layout/home_reading_list_page.xml \
|
||||||
res/layout/home_search_item_row.xml \
|
res/layout/home_search_item_row.xml \
|
||||||
res/layout/home_banner.xml \
|
res/layout/home_banner.xml \
|
||||||
res/layout/home_suggestion_prompt.xml \
|
res/layout/home_suggestion_prompt.xml \
|
||||||
|
res/layout/home_top_sites_page.xml \
|
||||||
res/layout/web_app.xml \
|
res/layout/web_app.xml \
|
||||||
res/layout/launch_app_list.xml \
|
res/layout/launch_app_list.xml \
|
||||||
res/layout/launch_app_listitem.xml \
|
res/layout/launch_app_listitem.xml \
|
||||||
|
@ -93,7 +93,7 @@ public class Tab {
|
|||||||
mUserSearch = "";
|
mUserSearch = "";
|
||||||
mExternal = external;
|
mExternal = external;
|
||||||
mParentId = parentId;
|
mParentId = parentId;
|
||||||
mAboutHomePage = HomePager.Page.BOOKMARKS;
|
mAboutHomePage = HomePager.Page.TOP_SITES;
|
||||||
mTitle = title == null ? "" : title;
|
mTitle = title == null ? "" : title;
|
||||||
mFavicon = null;
|
mFavicon = null;
|
||||||
mFaviconUrl = null;
|
mFaviconUrl = null;
|
||||||
|
@ -8,21 +8,12 @@ package org.mozilla.gecko.home;
|
|||||||
import org.mozilla.gecko.favicons.Favicons;
|
import org.mozilla.gecko.favicons.Favicons;
|
||||||
import org.mozilla.gecko.R;
|
import org.mozilla.gecko.R;
|
||||||
import org.mozilla.gecko.Tabs;
|
import org.mozilla.gecko.Tabs;
|
||||||
import org.mozilla.gecko.animation.PropertyAnimator;
|
|
||||||
import org.mozilla.gecko.animation.PropertyAnimator.Property;
|
|
||||||
import org.mozilla.gecko.animation.ViewHelper;
|
|
||||||
import org.mozilla.gecko.db.BrowserContract.Bookmarks;
|
import org.mozilla.gecko.db.BrowserContract.Bookmarks;
|
||||||
import org.mozilla.gecko.db.BrowserContract.Thumbnails;
|
|
||||||
import org.mozilla.gecko.db.BrowserDB;
|
import org.mozilla.gecko.db.BrowserDB;
|
||||||
import org.mozilla.gecko.db.BrowserDB.URLColumns;
|
import org.mozilla.gecko.db.BrowserDB.URLColumns;
|
||||||
import org.mozilla.gecko.gfx.BitmapUtils;
|
import org.mozilla.gecko.gfx.BitmapUtils;
|
||||||
import org.mozilla.gecko.home.BookmarksListAdapter.OnRefreshFolderListener;
|
import org.mozilla.gecko.home.BookmarksListAdapter.OnRefreshFolderListener;
|
||||||
import org.mozilla.gecko.home.HomeListView.HomeContextMenuInfo;
|
|
||||||
import org.mozilla.gecko.home.HomePager.OnUrlOpenListener;
|
import org.mozilla.gecko.home.HomePager.OnUrlOpenListener;
|
||||||
import org.mozilla.gecko.home.PinBookmarkDialog.OnBookmarkSelectedListener;
|
|
||||||
import org.mozilla.gecko.home.TopBookmarksAdapter.Thumbnail;
|
|
||||||
import org.mozilla.gecko.home.TopBookmarksView.OnPinBookmarkListener;
|
|
||||||
import org.mozilla.gecko.home.TopBookmarksView.TopBookmarksContextMenuInfo;
|
|
||||||
import org.mozilla.gecko.util.ThreadUtils;
|
import org.mozilla.gecko.util.ThreadUtils;
|
||||||
|
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
@ -32,27 +23,14 @@ import android.content.res.Configuration;
|
|||||||
import android.database.Cursor;
|
import android.database.Cursor;
|
||||||
import android.graphics.Bitmap;
|
import android.graphics.Bitmap;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.support.v4.app.FragmentManager;
|
|
||||||
import android.support.v4.app.LoaderManager;
|
import android.support.v4.app.LoaderManager;
|
||||||
import android.support.v4.app.LoaderManager.LoaderCallbacks;
|
import android.support.v4.app.LoaderManager.LoaderCallbacks;
|
||||||
import android.support.v4.content.AsyncTaskLoader;
|
|
||||||
import android.support.v4.content.Loader;
|
import android.support.v4.content.Loader;
|
||||||
import android.text.TextUtils;
|
import android.text.TextUtils;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.view.ContextMenu;
|
|
||||||
import android.view.ContextMenu.ContextMenuInfo;
|
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
import android.view.MenuInflater;
|
|
||||||
import android.view.MenuItem;
|
|
||||||
import android.view.MotionEvent;
|
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.view.View.OnTouchListener;
|
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
import android.widget.Toast;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A page in about:home that displays a ListView of bookmarks.
|
* A page in about:home that displays a ListView of bookmarks.
|
||||||
@ -63,57 +41,24 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
// Cursor loader ID for list of bookmarks.
|
// Cursor loader ID for list of bookmarks.
|
||||||
private static final int LOADER_ID_BOOKMARKS_LIST = 0;
|
private static final int LOADER_ID_BOOKMARKS_LIST = 0;
|
||||||
|
|
||||||
// Cursor loader ID for grid of bookmarks.
|
|
||||||
private static final int LOADER_ID_TOP_BOOKMARKS = 1;
|
|
||||||
|
|
||||||
// Loader ID for thumbnails.
|
|
||||||
private static final int LOADER_ID_THUMBNAILS = 2;
|
|
||||||
|
|
||||||
// Key for bookmarks folder id.
|
// Key for bookmarks folder id.
|
||||||
private static final String BOOKMARKS_FOLDER_KEY = "folder_id";
|
private static final String BOOKMARKS_FOLDER_KEY = "folder_id";
|
||||||
|
|
||||||
// Key for thumbnail urls.
|
|
||||||
private static final String THUMBNAILS_URLS_KEY = "urls";
|
|
||||||
|
|
||||||
// List of bookmarks.
|
// List of bookmarks.
|
||||||
private BookmarksListView mList;
|
private BookmarksListView mList;
|
||||||
|
|
||||||
// Grid of top bookmarks.
|
|
||||||
private TopBookmarksView mTopBookmarks;
|
|
||||||
|
|
||||||
// Banner to show snippets.
|
|
||||||
private HomeBanner mBanner;
|
|
||||||
|
|
||||||
// Adapter for list of bookmarks.
|
// Adapter for list of bookmarks.
|
||||||
private BookmarksListAdapter mListAdapter;
|
private BookmarksListAdapter mListAdapter;
|
||||||
|
|
||||||
// Adapter for grid of bookmarks.
|
|
||||||
private TopBookmarksAdapter mTopBookmarksAdapter;
|
|
||||||
|
|
||||||
// Callback for cursor loaders.
|
// Callback for cursor loaders.
|
||||||
private CursorLoaderCallbacks mLoaderCallbacks;
|
private CursorLoaderCallbacks mLoaderCallbacks;
|
||||||
|
|
||||||
// Callback for thumbnail loader.
|
|
||||||
private ThumbnailsLoaderCallbacks mThumbnailsLoaderCallbacks;
|
|
||||||
|
|
||||||
// Listener for pinning bookmarks.
|
|
||||||
private PinBookmarkListener mPinBookmarkListener;
|
|
||||||
|
|
||||||
// Raw Y value of the last event that happened on the list view.
|
|
||||||
private float mListTouchY = -1;
|
|
||||||
|
|
||||||
// Scrolling direction of the banner.
|
|
||||||
private boolean mSnapBannerToTop;
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
||||||
final View view = inflater.inflate(R.layout.home_bookmarks_page, container, false);
|
final View view = inflater.inflate(R.layout.home_bookmarks_page, container, false);
|
||||||
|
|
||||||
mList = (BookmarksListView) view.findViewById(R.id.bookmarks_list);
|
mList = (BookmarksListView) view.findViewById(R.id.bookmarks_list);
|
||||||
|
|
||||||
mTopBookmarks = new TopBookmarksView(getActivity());
|
|
||||||
mList.addHeaderView(mTopBookmarks);
|
|
||||||
|
|
||||||
return view;
|
return view;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -129,26 +74,11 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
+ " must implement HomePager.OnUrlOpenListener");
|
+ " must implement HomePager.OnUrlOpenListener");
|
||||||
}
|
}
|
||||||
|
|
||||||
mPinBookmarkListener = new PinBookmarkListener();
|
|
||||||
|
|
||||||
mList.setTag(HomePager.LIST_TAG_BOOKMARKS);
|
mList.setTag(HomePager.LIST_TAG_BOOKMARKS);
|
||||||
mList.setOnUrlOpenListener(listener);
|
mList.setOnUrlOpenListener(listener);
|
||||||
mList.setHeaderDividersEnabled(false);
|
mList.setHeaderDividersEnabled(false);
|
||||||
|
|
||||||
mTopBookmarks.setOnUrlOpenListener(listener);
|
|
||||||
mTopBookmarks.setOnPinBookmarkListener(mPinBookmarkListener);
|
|
||||||
|
|
||||||
registerForContextMenu(mList);
|
registerForContextMenu(mList);
|
||||||
registerForContextMenu(mTopBookmarks);
|
|
||||||
|
|
||||||
mBanner = (HomeBanner) view.findViewById(R.id.home_banner);
|
|
||||||
mList.setOnTouchListener(new OnTouchListener() {
|
|
||||||
@Override
|
|
||||||
public boolean onTouch(View v, MotionEvent event) {
|
|
||||||
BookmarksPage.this.handleListTouchEvent(event);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -157,10 +87,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
|
|
||||||
final Activity activity = getActivity();
|
final Activity activity = getActivity();
|
||||||
|
|
||||||
// Setup the top bookmarks adapter.
|
|
||||||
mTopBookmarksAdapter = new TopBookmarksAdapter(activity, null);
|
|
||||||
mTopBookmarks.setAdapter(mTopBookmarksAdapter);
|
|
||||||
|
|
||||||
// Setup the list adapter.
|
// Setup the list adapter.
|
||||||
mListAdapter = new BookmarksListAdapter(activity, null);
|
mListAdapter = new BookmarksListAdapter(activity, null);
|
||||||
mListAdapter.setOnRefreshFolderListener(new OnRefreshFolderListener() {
|
mListAdapter.setOnRefreshFolderListener(new OnRefreshFolderListener() {
|
||||||
@ -180,7 +106,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
|
|
||||||
// Create callbacks before the initial loader is started.
|
// Create callbacks before the initial loader is started.
|
||||||
mLoaderCallbacks = new CursorLoaderCallbacks(activity, getLoaderManager());
|
mLoaderCallbacks = new CursorLoaderCallbacks(activity, getLoaderManager());
|
||||||
mThumbnailsLoaderCallbacks = new ThumbnailsLoaderCallbacks();
|
|
||||||
loadIfVisible();
|
loadIfVisible();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -188,9 +113,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
public void onDestroyView() {
|
public void onDestroyView() {
|
||||||
mList = null;
|
mList = null;
|
||||||
mListAdapter = null;
|
mListAdapter = null;
|
||||||
mTopBookmarks = null;
|
|
||||||
mTopBookmarksAdapter = null;
|
|
||||||
mPinBookmarkListener = null;
|
|
||||||
super.onDestroyView();
|
super.onDestroyView();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -214,204 +136,9 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void handleListTouchEvent(MotionEvent event) {
|
|
||||||
// Ignore the event if the banner is hidden for this session.
|
|
||||||
if (mBanner.isDismissed()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (event.getActionMasked()) {
|
|
||||||
case MotionEvent.ACTION_DOWN: {
|
|
||||||
mListTouchY = event.getRawY();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_MOVE: {
|
|
||||||
// There is a chance that we won't receive ACTION_DOWN, if the touch event
|
|
||||||
// actually started on the Grid instead of the List. Treat this as first event.
|
|
||||||
if (mListTouchY == -1) {
|
|
||||||
mListTouchY = event.getRawY();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final float curY = event.getRawY();
|
|
||||||
final float delta = mListTouchY - curY;
|
|
||||||
mSnapBannerToTop = (delta > 0.0f) ? false : true;
|
|
||||||
|
|
||||||
final float height = mBanner.getHeight();
|
|
||||||
float newTranslationY = ViewHelper.getTranslationY(mBanner) + delta;
|
|
||||||
|
|
||||||
// Clamp the values to be between 0 and height.
|
|
||||||
if (newTranslationY < 0.0f) {
|
|
||||||
newTranslationY = 0.0f;
|
|
||||||
} else if (newTranslationY > height) {
|
|
||||||
newTranslationY = height;
|
|
||||||
}
|
|
||||||
|
|
||||||
ViewHelper.setTranslationY(mBanner, newTranslationY);
|
|
||||||
mListTouchY = curY;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_UP:
|
|
||||||
case MotionEvent.ACTION_CANCEL: {
|
|
||||||
mListTouchY = -1;
|
|
||||||
final float y = ViewHelper.getTranslationY(mBanner);
|
|
||||||
final float height = mBanner.getHeight();
|
|
||||||
if (y > 0.0f && y < height) {
|
|
||||||
final PropertyAnimator animator = new PropertyAnimator(100);
|
|
||||||
animator.attach(mBanner, Property.TRANSLATION_Y, mSnapBannerToTop ? 0 : height);
|
|
||||||
animator.start();
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreateContextMenu(ContextMenu menu, View view, ContextMenuInfo menuInfo) {
|
|
||||||
if (menuInfo == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// HomeFragment will handle the default case.
|
|
||||||
if (menuInfo instanceof HomeContextMenuInfo) {
|
|
||||||
super.onCreateContextMenu(menu, view, menuInfo);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!(menuInfo instanceof TopBookmarksContextMenuInfo)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
MenuInflater inflater = new MenuInflater(view.getContext());
|
|
||||||
inflater.inflate(R.menu.top_bookmarks_contextmenu, menu);
|
|
||||||
|
|
||||||
TopBookmarksContextMenuInfo info = (TopBookmarksContextMenuInfo) menuInfo;
|
|
||||||
menu.setHeaderTitle(info.getDisplayTitle());
|
|
||||||
|
|
||||||
if (!TextUtils.isEmpty(info.url)) {
|
|
||||||
if (info.isPinned) {
|
|
||||||
menu.findItem(R.id.top_bookmarks_pin).setVisible(false);
|
|
||||||
} else {
|
|
||||||
menu.findItem(R.id.top_bookmarks_unpin).setVisible(false);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
menu.findItem(R.id.top_bookmarks_open_new_tab).setVisible(false);
|
|
||||||
menu.findItem(R.id.top_bookmarks_open_private_tab).setVisible(false);
|
|
||||||
menu.findItem(R.id.top_bookmarks_pin).setVisible(false);
|
|
||||||
menu.findItem(R.id.top_bookmarks_unpin).setVisible(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onContextItemSelected(MenuItem item) {
|
|
||||||
ContextMenuInfo menuInfo = item.getMenuInfo();
|
|
||||||
|
|
||||||
// HomeFragment will handle the default case.
|
|
||||||
if (menuInfo == null || !(menuInfo instanceof TopBookmarksContextMenuInfo)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
TopBookmarksContextMenuInfo info = (TopBookmarksContextMenuInfo) menuInfo;
|
|
||||||
final Activity activity = getActivity();
|
|
||||||
|
|
||||||
final int itemId = item.getItemId();
|
|
||||||
if (itemId == R.id.top_bookmarks_open_new_tab || itemId == R.id.top_bookmarks_open_private_tab) {
|
|
||||||
if (info.url == null) {
|
|
||||||
Log.e(LOGTAG, "Can't open in new tab because URL is null");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
int flags = Tabs.LOADURL_NEW_TAB | Tabs.LOADURL_BACKGROUND;
|
|
||||||
if (item.getItemId() == R.id.top_bookmarks_open_private_tab)
|
|
||||||
flags |= Tabs.LOADURL_PRIVATE;
|
|
||||||
|
|
||||||
Tabs.getInstance().loadUrl(info.url, flags);
|
|
||||||
Toast.makeText(activity, R.string.new_tab_opened, Toast.LENGTH_SHORT).show();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (itemId == R.id.top_bookmarks_pin) {
|
|
||||||
final String url = info.url;
|
|
||||||
final String title = info.title;
|
|
||||||
final int position = info.position;
|
|
||||||
final Context context = getActivity().getApplicationContext();
|
|
||||||
|
|
||||||
ThreadUtils.postToBackgroundThread(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
BrowserDB.pinSite(context.getContentResolver(), url, title, position);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (itemId == R.id.top_bookmarks_unpin) {
|
|
||||||
final int position = info.position;
|
|
||||||
final Context context = getActivity().getApplicationContext();
|
|
||||||
|
|
||||||
ThreadUtils.postToBackgroundThread(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
BrowserDB.unpinSite(context.getContentResolver(), position);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (itemId == R.id.top_bookmarks_edit) {
|
|
||||||
mPinBookmarkListener.onPinBookmark(info.position);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void load() {
|
protected void load() {
|
||||||
final LoaderManager manager = getLoaderManager();
|
getLoaderManager().initLoader(LOADER_ID_BOOKMARKS_LIST, null, mLoaderCallbacks);
|
||||||
manager.initLoader(LOADER_ID_BOOKMARKS_LIST, null, mLoaderCallbacks);
|
|
||||||
manager.initLoader(LOADER_ID_TOP_BOOKMARKS, null, mLoaderCallbacks);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Listener for pinning bookmarks.
|
|
||||||
*/
|
|
||||||
private class PinBookmarkListener implements OnPinBookmarkListener,
|
|
||||||
OnBookmarkSelectedListener {
|
|
||||||
// Tag for the PinBookmarkDialog fragment.
|
|
||||||
private static final String TAG_PIN_BOOKMARK = "pin_bookmark";
|
|
||||||
|
|
||||||
// Position of the pin.
|
|
||||||
private int mPosition;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onPinBookmark(int position) {
|
|
||||||
mPosition = position;
|
|
||||||
|
|
||||||
final FragmentManager manager = getActivity().getSupportFragmentManager();
|
|
||||||
PinBookmarkDialog dialog = (PinBookmarkDialog) manager.findFragmentByTag(TAG_PIN_BOOKMARK);
|
|
||||||
if (dialog == null) {
|
|
||||||
dialog = PinBookmarkDialog.newInstance();
|
|
||||||
}
|
|
||||||
|
|
||||||
dialog.setOnBookmarkSelectedListener(this);
|
|
||||||
dialog.show(manager, TAG_PIN_BOOKMARK);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onBookmarkSelected(final String url, final String title) {
|
|
||||||
final int position = mPosition;
|
|
||||||
final Context context = getActivity().getApplicationContext();
|
|
||||||
ThreadUtils.postToBackgroundThread(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
BrowserDB.pinSite(context.getContentResolver(), url, title, position);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -435,21 +162,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Loader for the grid for top bookmarks.
|
|
||||||
*/
|
|
||||||
private static class TopBookmarksLoader extends SimpleCursorLoader {
|
|
||||||
public TopBookmarksLoader(Context context) {
|
|
||||||
super(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Cursor loadCursor() {
|
|
||||||
final int max = getContext().getResources().getInteger(R.integer.number_of_top_sites);
|
|
||||||
return BrowserDB.getTopBookmarks(getContext().getContentResolver(), max);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loader callbacks for the LoaderManager of this fragment.
|
* Loader callbacks for the LoaderManager of this fragment.
|
||||||
*/
|
*/
|
||||||
@ -469,10 +181,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case LOADER_ID_TOP_BOOKMARKS: {
|
|
||||||
return new TopBookmarksLoader(getActivity());
|
|
||||||
}
|
|
||||||
|
|
||||||
default: {
|
default: {
|
||||||
return super.onCreateLoader(id, args);
|
return super.onCreateLoader(id, args);
|
||||||
}
|
}
|
||||||
@ -490,26 +198,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case LOADER_ID_TOP_BOOKMARKS: {
|
|
||||||
mTopBookmarksAdapter.swapCursor(c);
|
|
||||||
|
|
||||||
// Load the thumbnails.
|
|
||||||
if (c.getCount() > 0 && c.moveToFirst()) {
|
|
||||||
final ArrayList<String> urls = new ArrayList<String>();
|
|
||||||
do {
|
|
||||||
final String url = c.getString(c.getColumnIndexOrThrow(URLColumns.URL));
|
|
||||||
urls.add(url);
|
|
||||||
} while (c.moveToNext());
|
|
||||||
|
|
||||||
if (urls.size() > 0) {
|
|
||||||
Bundle bundle = new Bundle();
|
|
||||||
bundle.putStringArrayList(THUMBNAILS_URLS_KEY, urls);
|
|
||||||
getLoaderManager().restartLoader(LOADER_ID_THUMBNAILS, bundle, mThumbnailsLoaderCallbacks);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
default: {
|
default: {
|
||||||
super.onLoadFinished(loader, c);
|
super.onLoadFinished(loader, c);
|
||||||
break;
|
break;
|
||||||
@ -528,13 +216,6 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case LOADER_ID_TOP_BOOKMARKS: {
|
|
||||||
if (mTopBookmarks != null) {
|
|
||||||
mTopBookmarksAdapter.swapCursor(null);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
default: {
|
default: {
|
||||||
super.onLoaderReset(loader);
|
super.onLoaderReset(loader);
|
||||||
break;
|
break;
|
||||||
@ -547,132 +228,4 @@ public class BookmarksPage extends HomeFragment {
|
|||||||
mListAdapter.notifyDataSetChanged();
|
mListAdapter.notifyDataSetChanged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* An AsyncTaskLoader to load the thumbnails from a cursor.
|
|
||||||
*/
|
|
||||||
private static class ThumbnailsLoader extends AsyncTaskLoader<Map<String, Thumbnail>> {
|
|
||||||
private Map<String, Thumbnail> mThumbnails;
|
|
||||||
private ArrayList<String> mUrls;
|
|
||||||
|
|
||||||
public ThumbnailsLoader(Context context, ArrayList<String> urls) {
|
|
||||||
super(context);
|
|
||||||
mUrls = urls;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Map<String, Thumbnail> loadInBackground() {
|
|
||||||
if (mUrls == null || mUrls.size() == 0) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
final Map<String, Thumbnail> thumbnails = new HashMap<String, Thumbnail>();
|
|
||||||
|
|
||||||
// Query the DB for thumbnails.
|
|
||||||
final ContentResolver cr = getContext().getContentResolver();
|
|
||||||
final Cursor cursor = BrowserDB.getThumbnailsForUrls(cr, mUrls);
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (cursor != null && cursor.moveToFirst()) {
|
|
||||||
do {
|
|
||||||
// Try to get the thumbnail, if cursor is valid.
|
|
||||||
String url = cursor.getString(cursor.getColumnIndexOrThrow(Thumbnails.URL));
|
|
||||||
final byte[] b = cursor.getBlob(cursor.getColumnIndexOrThrow(Thumbnails.DATA));
|
|
||||||
final Bitmap bitmap = (b == null ? null : BitmapUtils.decodeByteArray(b));
|
|
||||||
|
|
||||||
if (bitmap != null) {
|
|
||||||
thumbnails.put(url, new Thumbnail(bitmap, true));
|
|
||||||
}
|
|
||||||
} while (cursor.moveToNext());
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
if (cursor != null) {
|
|
||||||
cursor.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Query the DB for favicons for the urls without thumbnails.
|
|
||||||
for (String url : mUrls) {
|
|
||||||
if (!thumbnails.containsKey(url)) {
|
|
||||||
final Bitmap bitmap = BrowserDB.getFaviconForUrl(cr, url);
|
|
||||||
if (bitmap != null) {
|
|
||||||
// Favicons.scaleImage can return several different size favicons,
|
|
||||||
// but will at least prevent this from being too large.
|
|
||||||
thumbnails.put(url, new Thumbnail(Favicons.scaleImage(bitmap), false));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return thumbnails;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void deliverResult(Map<String, Thumbnail> thumbnails) {
|
|
||||||
if (isReset()) {
|
|
||||||
mThumbnails = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mThumbnails = thumbnails;
|
|
||||||
|
|
||||||
if (isStarted()) {
|
|
||||||
super.deliverResult(thumbnails);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onStartLoading() {
|
|
||||||
if (mThumbnails != null) {
|
|
||||||
deliverResult(mThumbnails);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (takeContentChanged() || mThumbnails == null) {
|
|
||||||
forceLoad();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onStopLoading() {
|
|
||||||
cancelLoad();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCanceled(Map<String, Thumbnail> thumbnails) {
|
|
||||||
mThumbnails = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onReset() {
|
|
||||||
super.onReset();
|
|
||||||
|
|
||||||
// Ensure the loader is stopped.
|
|
||||||
onStopLoading();
|
|
||||||
|
|
||||||
mThumbnails = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Loader callbacks for the thumbnails on TopBookmarksView.
|
|
||||||
*/
|
|
||||||
private class ThumbnailsLoaderCallbacks implements LoaderCallbacks<Map<String, Thumbnail>> {
|
|
||||||
@Override
|
|
||||||
public Loader<Map<String, Thumbnail>> onCreateLoader(int id, Bundle args) {
|
|
||||||
return new ThumbnailsLoader(getActivity(), args.getStringArrayList(THUMBNAILS_URLS_KEY));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onLoadFinished(Loader<Map<String, Thumbnail>> loader, Map<String, Thumbnail> thumbnails) {
|
|
||||||
if (mTopBookmarksAdapter != null) {
|
|
||||||
mTopBookmarksAdapter.updateThumbnails(thumbnails);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onLoaderReset(Loader<Map<String, Thumbnail>> loader) {
|
|
||||||
if (mTopBookmarksAdapter != null) {
|
|
||||||
mTopBookmarksAdapter.updateThumbnails(null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -25,7 +25,7 @@ public class HistoryPage extends HomeFragment
|
|||||||
private static final String LOGTAG = "GeckoHistoryPage";
|
private static final String LOGTAG = "GeckoHistoryPage";
|
||||||
private IconTabWidget mTabWidget;
|
private IconTabWidget mTabWidget;
|
||||||
private int mSelectedTab;
|
private int mSelectedTab;
|
||||||
private boolean initializeVisitedPage;
|
private boolean initializeRecentPage;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
||||||
@ -38,7 +38,6 @@ public class HistoryPage extends HomeFragment
|
|||||||
|
|
||||||
mTabWidget = (IconTabWidget) view.findViewById(R.id.tab_icon_widget);
|
mTabWidget = (IconTabWidget) view.findViewById(R.id.tab_icon_widget);
|
||||||
|
|
||||||
mTabWidget.addTab(R.drawable.icon_most_visited, R.string.home_most_visited_title);
|
|
||||||
mTabWidget.addTab(R.drawable.icon_most_recent, R.string.home_most_recent_title);
|
mTabWidget.addTab(R.drawable.icon_most_recent, R.string.home_most_recent_title);
|
||||||
mTabWidget.addTab(R.drawable.icon_last_tabs, R.string.home_last_tabs_title);
|
mTabWidget.addTab(R.drawable.icon_last_tabs, R.string.home_last_tabs_title);
|
||||||
|
|
||||||
@ -50,11 +49,11 @@ public class HistoryPage extends HomeFragment
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void load() {
|
public void load() {
|
||||||
// Show most visited page as the initial page.
|
// Show most recent page as the initial page.
|
||||||
// Since we detach/attach on config change, this prevents from replacing current fragment.
|
// Since we detach/attach on config change, this prevents from replacing current fragment.
|
||||||
if (!initializeVisitedPage) {
|
if (!initializeRecentPage) {
|
||||||
showMostVisitedPage();
|
showMostRecentPage();
|
||||||
initializeVisitedPage = true;
|
initializeRecentPage = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -65,10 +64,8 @@ public class HistoryPage extends HomeFragment
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (index == 0) {
|
if (index == 0) {
|
||||||
showMostVisitedPage();
|
|
||||||
} else if (index == 1) {
|
|
||||||
showMostRecentPage();
|
showMostRecentPage();
|
||||||
} else if (index == 2) {
|
} else if (index == 1) {
|
||||||
showLastTabsPage();
|
showLastTabsPage();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,15 +92,10 @@ public class HistoryPage extends HomeFragment
|
|||||||
subPage.setArguments(args);
|
subPage.setArguments(args);
|
||||||
|
|
||||||
getChildFragmentManager().beginTransaction()
|
getChildFragmentManager().beginTransaction()
|
||||||
.addToBackStack(null).replace(R.id.visited_page_container, subPage)
|
.addToBackStack(null).replace(R.id.history_page_container, subPage)
|
||||||
.commitAllowingStateLoss();
|
.commitAllowingStateLoss();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void showMostVisitedPage() {
|
|
||||||
final MostVisitedPage mostVisitedPage = MostVisitedPage.newInstance();
|
|
||||||
showSubPage(mostVisitedPage);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void showMostRecentPage() {
|
private void showMostRecentPage() {
|
||||||
final MostRecentPage mostRecentPage = MostRecentPage.newInstance();
|
final MostRecentPage mostRecentPage = MostRecentPage.newInstance();
|
||||||
showSubPage(mostRecentPage);
|
showSubPage(mostRecentPage);
|
||||||
|
@ -8,6 +8,7 @@ package org.mozilla.gecko.home;
|
|||||||
import org.mozilla.gecko.R;
|
import org.mozilla.gecko.R;
|
||||||
import org.mozilla.gecko.animation.PropertyAnimator;
|
import org.mozilla.gecko.animation.PropertyAnimator;
|
||||||
import org.mozilla.gecko.animation.ViewHelper;
|
import org.mozilla.gecko.animation.ViewHelper;
|
||||||
|
import org.mozilla.gecko.util.HardwareUtils;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.os.Build;
|
import android.os.Build;
|
||||||
@ -38,6 +39,7 @@ public class HomePager extends ViewPager {
|
|||||||
// List of pages in order.
|
// List of pages in order.
|
||||||
public enum Page {
|
public enum Page {
|
||||||
HISTORY,
|
HISTORY,
|
||||||
|
TOP_SITES,
|
||||||
BOOKMARKS,
|
BOOKMARKS,
|
||||||
READING_LIST
|
READING_LIST
|
||||||
}
|
}
|
||||||
@ -90,9 +92,9 @@ public class HomePager extends ViewPager {
|
|||||||
super(context, attrs);
|
super(context, attrs);
|
||||||
mContext = context;
|
mContext = context;
|
||||||
|
|
||||||
// This is to keep all 3 pages in memory after they are
|
// This is to keep all 4 pages in memory after they are
|
||||||
// selected in the pager.
|
// selected in the pager.
|
||||||
setOffscreenPageLimit(2);
|
setOffscreenPageLimit(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -131,14 +133,19 @@ public class HomePager extends ViewPager {
|
|||||||
// Only animate on post-HC devices, when a non-null animator is given
|
// Only animate on post-HC devices, when a non-null animator is given
|
||||||
final boolean shouldAnimate = (animator != null && Build.VERSION.SDK_INT >= 11);
|
final boolean shouldAnimate = (animator != null && Build.VERSION.SDK_INT >= 11);
|
||||||
|
|
||||||
// Add the pages to the adapter in order.
|
adapter.addTab(Page.TOP_SITES, TopSitesPage.class, new Bundle(),
|
||||||
adapter.addTab(Page.HISTORY, HistoryPage.class, new Bundle(),
|
getContext().getString(R.string.home_top_sites_title));
|
||||||
getContext().getString(R.string.home_history_title));
|
|
||||||
adapter.addTab(Page.BOOKMARKS, BookmarksPage.class, new Bundle(),
|
adapter.addTab(Page.BOOKMARKS, BookmarksPage.class, new Bundle(),
|
||||||
getContext().getString(R.string.bookmarks_title));
|
getContext().getString(R.string.bookmarks_title));
|
||||||
adapter.addTab(Page.READING_LIST, ReadingListPage.class, new Bundle(),
|
adapter.addTab(Page.READING_LIST, ReadingListPage.class, new Bundle(),
|
||||||
getContext().getString(R.string.reading_list_title));
|
getContext().getString(R.string.reading_list_title));
|
||||||
|
|
||||||
|
// On phones, the history tab is the first tab. On tablets, the
|
||||||
|
// history tab is the last tab.
|
||||||
|
adapter.addTab(HardwareUtils.isTablet() ? -1 : 0,
|
||||||
|
Page.HISTORY, HistoryPage.class, new Bundle(),
|
||||||
|
getContext().getString(R.string.home_history_title));
|
||||||
|
|
||||||
adapter.setCanLoadHint(!shouldAnimate);
|
adapter.setCanLoadHint(!shouldAnimate);
|
||||||
|
|
||||||
setAdapter(adapter);
|
setAdapter(adapter);
|
||||||
@ -226,8 +233,18 @@ public class HomePager extends ViewPager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public void addTab(Page page, Class<?> clss, Bundle args, String title) {
|
public void addTab(Page page, Class<?> clss, Bundle args, String title) {
|
||||||
|
addTab(-1, page, clss, args, title);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addTab(int index, Page page, Class<?> clss, Bundle args, String title) {
|
||||||
TabInfo info = new TabInfo(page, clss, args, title);
|
TabInfo info = new TabInfo(page, clss, args, title);
|
||||||
mTabs.add(info);
|
|
||||||
|
if (index >= 0) {
|
||||||
|
mTabs.add(index, info);
|
||||||
|
} else {
|
||||||
|
mTabs.add(info);
|
||||||
|
}
|
||||||
|
|
||||||
notifyDataSetChanged();
|
notifyDataSetChanged();
|
||||||
|
|
||||||
if (mDecor != null) {
|
if (mDecor != null) {
|
||||||
|
@ -6,6 +6,9 @@
|
|||||||
package org.mozilla.gecko.home;
|
package org.mozilla.gecko.home;
|
||||||
|
|
||||||
import org.mozilla.gecko.R;
|
import org.mozilla.gecko.R;
|
||||||
|
import org.mozilla.gecko.animation.PropertyAnimator;
|
||||||
|
import org.mozilla.gecko.animation.PropertyAnimator.Property;
|
||||||
|
import org.mozilla.gecko.animation.ViewHelper;
|
||||||
import org.mozilla.gecko.db.BrowserDB;
|
import org.mozilla.gecko.db.BrowserDB;
|
||||||
import org.mozilla.gecko.db.BrowserDB.URLColumns;
|
import org.mozilla.gecko.db.BrowserDB.URLColumns;
|
||||||
import org.mozilla.gecko.home.HomePager.OnUrlOpenListener;
|
import org.mozilla.gecko.home.HomePager.OnUrlOpenListener;
|
||||||
@ -13,13 +16,16 @@ import org.mozilla.gecko.home.HomePager.OnUrlOpenListener;
|
|||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.content.ContentResolver;
|
import android.content.ContentResolver;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
|
import android.content.res.Configuration;
|
||||||
import android.database.Cursor;
|
import android.database.Cursor;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
import android.support.v4.app.LoaderManager;
|
import android.support.v4.app.LoaderManager;
|
||||||
import android.support.v4.content.Loader;
|
import android.support.v4.content.Loader;
|
||||||
import android.support.v4.widget.CursorAdapter;
|
import android.support.v4.widget.CursorAdapter;
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.MotionEvent;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
|
import android.view.View.OnTouchListener;
|
||||||
import android.view.ViewGroup;
|
import android.view.ViewGroup;
|
||||||
import android.view.ViewStub;
|
import android.view.ViewStub;
|
||||||
import android.widget.AdapterView;
|
import android.widget.AdapterView;
|
||||||
@ -32,12 +38,12 @@ import java.util.EnumSet;
|
|||||||
/**
|
/**
|
||||||
* Fragment that displays frecency search results in a ListView.
|
* Fragment that displays frecency search results in a ListView.
|
||||||
*/
|
*/
|
||||||
public class MostVisitedPage extends HomeFragment {
|
public class TopSitesPage extends HomeFragment {
|
||||||
// Logging tag name
|
// Logging tag name
|
||||||
private static final String LOGTAG = "GeckoMostVisitedPage";
|
private static final String LOGTAG = "GeckoTopSitesPage";
|
||||||
|
|
||||||
// Cursor loader ID for search query
|
// Cursor loader ID for search query
|
||||||
private static final int LOADER_ID_FRECENCY = 0;
|
private static final int LOADER_ID_TOP_SITES = 0;
|
||||||
|
|
||||||
// Adapter for the list of search results
|
// Adapter for the list of search results
|
||||||
private VisitedAdapter mAdapter;
|
private VisitedAdapter mAdapter;
|
||||||
@ -45,23 +51,29 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
// The view shown by the fragment.
|
// The view shown by the fragment.
|
||||||
private ListView mList;
|
private ListView mList;
|
||||||
|
|
||||||
// The title for this HomeFragment page.
|
|
||||||
private TextView mTitle;
|
|
||||||
|
|
||||||
// Reference to the View to display when there are no results.
|
// Reference to the View to display when there are no results.
|
||||||
private View mEmptyView;
|
private View mEmptyView;
|
||||||
|
|
||||||
|
// Banner to show snippets.
|
||||||
|
private HomeBanner mBanner;
|
||||||
|
|
||||||
|
// Raw Y value of the last event that happened on the list view.
|
||||||
|
private float mListTouchY = -1;
|
||||||
|
|
||||||
|
// Scrolling direction of the banner.
|
||||||
|
private boolean mSnapBannerToTop;
|
||||||
|
|
||||||
// Callbacks used for the search and favicon cursor loaders
|
// Callbacks used for the search and favicon cursor loaders
|
||||||
private CursorLoaderCallbacks mCursorLoaderCallbacks;
|
private CursorLoaderCallbacks mCursorLoaderCallbacks;
|
||||||
|
|
||||||
// On URL open listener
|
// On URL open listener
|
||||||
private OnUrlOpenListener mUrlOpenListener;
|
private OnUrlOpenListener mUrlOpenListener;
|
||||||
|
|
||||||
public static MostVisitedPage newInstance() {
|
public static TopSitesPage newInstance() {
|
||||||
return new MostVisitedPage();
|
return new TopSitesPage();
|
||||||
}
|
}
|
||||||
|
|
||||||
public MostVisitedPage() {
|
public TopSitesPage() {
|
||||||
mUrlOpenListener = null;
|
mUrlOpenListener = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -86,16 +98,11 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
||||||
return inflater.inflate(R.layout.home_most_visited_page, container, false);
|
return inflater.inflate(R.layout.home_top_sites_page, container, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onViewCreated(View view, Bundle savedInstanceState) {
|
public void onViewCreated(View view, Bundle savedInstanceState) {
|
||||||
mTitle = (TextView) view.findViewById(R.id.title);
|
|
||||||
if (mTitle != null) {
|
|
||||||
mTitle.setText(R.string.home_most_visited_title);
|
|
||||||
}
|
|
||||||
|
|
||||||
mList = (HomeListView) view.findViewById(R.id.list);
|
mList = (HomeListView) view.findViewById(R.id.list);
|
||||||
mList.setTag(HomePager.LIST_TAG_MOST_VISITED);
|
mList.setTag(HomePager.LIST_TAG_MOST_VISITED);
|
||||||
|
|
||||||
@ -115,16 +122,37 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
});
|
});
|
||||||
|
|
||||||
registerForContextMenu(mList);
|
registerForContextMenu(mList);
|
||||||
|
|
||||||
|
mBanner = (HomeBanner) view.findViewById(R.id.home_banner);
|
||||||
|
mList.setOnTouchListener(new OnTouchListener() {
|
||||||
|
@Override
|
||||||
|
public boolean onTouch(View v, MotionEvent event) {
|
||||||
|
TopSitesPage.this.handleListTouchEvent(event);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onDestroyView() {
|
public void onDestroyView() {
|
||||||
super.onDestroyView();
|
super.onDestroyView();
|
||||||
mList = null;
|
mList = null;
|
||||||
mTitle = null;
|
|
||||||
mEmptyView = null;
|
mEmptyView = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onConfigurationChanged(Configuration newConfig) {
|
||||||
|
super.onConfigurationChanged(newConfig);
|
||||||
|
|
||||||
|
// Detach and reattach the fragment as the layout changes.
|
||||||
|
if (isVisible()) {
|
||||||
|
getFragmentManager().beginTransaction()
|
||||||
|
.detach(this)
|
||||||
|
.attach(this)
|
||||||
|
.commitAllowingStateLoss();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onActivityCreated(Bundle savedInstanceState) {
|
public void onActivityCreated(Bundle savedInstanceState) {
|
||||||
super.onActivityCreated(savedInstanceState);
|
super.onActivityCreated(savedInstanceState);
|
||||||
@ -142,23 +170,68 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void load() {
|
protected void load() {
|
||||||
getLoaderManager().initLoader(LOADER_ID_FRECENCY, null, mCursorLoaderCallbacks);
|
getLoaderManager().initLoader(LOADER_ID_TOP_SITES, null, mCursorLoaderCallbacks);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleListTouchEvent(MotionEvent event) {
|
||||||
|
// Ignore the event if the banner is hidden for this session.
|
||||||
|
if (mBanner.isDismissed()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (event.getActionMasked()) {
|
||||||
|
case MotionEvent.ACTION_DOWN: {
|
||||||
|
mListTouchY = event.getRawY();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MotionEvent.ACTION_MOVE: {
|
||||||
|
// There is a chance that we won't receive ACTION_DOWN, if the touch event
|
||||||
|
// actually started on the Grid instead of the List. Treat this as first event.
|
||||||
|
if (mListTouchY == -1) {
|
||||||
|
mListTouchY = event.getRawY();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final float curY = event.getRawY();
|
||||||
|
final float delta = mListTouchY - curY;
|
||||||
|
mSnapBannerToTop = (delta > 0.0f) ? false : true;
|
||||||
|
|
||||||
|
final float height = mBanner.getHeight();
|
||||||
|
float newTranslationY = ViewHelper.getTranslationY(mBanner) + delta;
|
||||||
|
|
||||||
|
// Clamp the values to be between 0 and height.
|
||||||
|
if (newTranslationY < 0.0f) {
|
||||||
|
newTranslationY = 0.0f;
|
||||||
|
} else if (newTranslationY > height) {
|
||||||
|
newTranslationY = height;
|
||||||
|
}
|
||||||
|
|
||||||
|
ViewHelper.setTranslationY(mBanner, newTranslationY);
|
||||||
|
mListTouchY = curY;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case MotionEvent.ACTION_UP:
|
||||||
|
case MotionEvent.ACTION_CANCEL: {
|
||||||
|
mListTouchY = -1;
|
||||||
|
final float y = ViewHelper.getTranslationY(mBanner);
|
||||||
|
final float height = mBanner.getHeight();
|
||||||
|
if (y > 0.0f && y < height) {
|
||||||
|
final PropertyAnimator animator = new PropertyAnimator(100);
|
||||||
|
animator.attach(mBanner, Property.TRANSLATION_Y, mSnapBannerToTop ? 0 : height);
|
||||||
|
animator.start();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void updateUiFromCursor(Cursor c) {
|
private void updateUiFromCursor(Cursor c) {
|
||||||
if (c != null && c.getCount() > 0) {
|
if (c != null && c.getCount() > 0) {
|
||||||
if (mTitle != null) {
|
|
||||||
mTitle.setVisibility(View.VISIBLE);
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cursor is empty, so hide the title and set the
|
|
||||||
// empty view if it hasn't been set already.
|
|
||||||
if (mTitle != null) {
|
|
||||||
mTitle.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mEmptyView == null) {
|
if (mEmptyView == null) {
|
||||||
// Set empty page view. We delay this so that the empty view won't flash.
|
// Set empty page view. We delay this so that the empty view won't flash.
|
||||||
ViewStub emptyViewStub = (ViewStub) getView().findViewById(R.id.home_empty_view_stub);
|
ViewStub emptyViewStub = (ViewStub) getView().findViewById(R.id.home_empty_view_stub);
|
||||||
@ -174,11 +247,11 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class FrecencyCursorLoader extends SimpleCursorLoader {
|
private static class TopSitesCursorLoader extends SimpleCursorLoader {
|
||||||
// Max number of search results
|
// Max number of search results
|
||||||
private static final int SEARCH_LIMIT = 50;
|
private static final int SEARCH_LIMIT = 50;
|
||||||
|
|
||||||
public FrecencyCursorLoader(Context context) {
|
public TopSitesCursorLoader(Context context) {
|
||||||
super(context);
|
super(context);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -213,8 +286,8 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
|
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
|
||||||
if (id == LOADER_ID_FRECENCY) {
|
if (id == LOADER_ID_TOP_SITES) {
|
||||||
return new FrecencyCursorLoader(getActivity());
|
return new TopSitesCursorLoader(getActivity());
|
||||||
} else {
|
} else {
|
||||||
return super.onCreateLoader(id, args);
|
return super.onCreateLoader(id, args);
|
||||||
}
|
}
|
||||||
@ -222,7 +295,7 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onLoadFinished(Loader<Cursor> loader, Cursor c) {
|
public void onLoadFinished(Loader<Cursor> loader, Cursor c) {
|
||||||
if (loader.getId() == LOADER_ID_FRECENCY) {
|
if (loader.getId() == LOADER_ID_TOP_SITES) {
|
||||||
mAdapter.swapCursor(c);
|
mAdapter.swapCursor(c);
|
||||||
updateUiFromCursor(c);
|
updateUiFromCursor(c);
|
||||||
loadFavicons(c);
|
loadFavicons(c);
|
||||||
@ -233,7 +306,7 @@ public class MostVisitedPage extends HomeFragment {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onLoaderReset(Loader<Cursor> loader) {
|
public void onLoaderReset(Loader<Cursor> loader) {
|
||||||
if (loader.getId() == LOADER_ID_FRECENCY) {
|
if (loader.getId() == LOADER_ID_TOP_SITES) {
|
||||||
mAdapter.swapCursor(null);
|
mAdapter.swapCursor(null);
|
||||||
} else {
|
} else {
|
||||||
super.onLoaderReset(loader);
|
super.onLoaderReset(loader);
|
@ -270,6 +270,7 @@ size. -->
|
|||||||
<!ENTITY button_set "Set">
|
<!ENTITY button_set "Set">
|
||||||
<!ENTITY button_clear "Clear">
|
<!ENTITY button_clear "Clear">
|
||||||
|
|
||||||
|
<!ENTITY home_top_sites_title "Top Sites">
|
||||||
<!ENTITY home_history_title "History">
|
<!ENTITY home_history_title "History">
|
||||||
<!ENTITY home_last_tabs_title "Tabs from last time">
|
<!ENTITY home_last_tabs_title "Tabs from last time">
|
||||||
<!ENTITY home_last_tabs_open "Open all tabs from last time">
|
<!ENTITY home_last_tabs_open "Open all tabs from last time">
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
android:layout="@layout/home_history_tabs_indicator"
|
android:layout="@layout/home_history_tabs_indicator"
|
||||||
gecko:display="text"/>
|
gecko:display="text"/>
|
||||||
|
|
||||||
<FrameLayout android:id="@+id/visited_page_container"
|
<FrameLayout android:id="@+id/history_page_container"
|
||||||
android:layout_width="0dp"
|
android:layout_width="0dp"
|
||||||
android:layout_height="fill_parent"
|
android:layout_height="fill_parent"
|
||||||
android:layout_weight="1" />
|
android:layout_weight="1" />
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
android:layout="@layout/home_history_tabs_indicator"
|
android:layout="@layout/home_history_tabs_indicator"
|
||||||
gecko:display="text"/>
|
gecko:display="text"/>
|
||||||
|
|
||||||
<FrameLayout android:id="@+id/visited_page_container"
|
<FrameLayout android:id="@+id/history_page_container"
|
||||||
android:layout_width="0dp"
|
android:layout_width="0dp"
|
||||||
android:layout_height="fill_parent"
|
android:layout_height="fill_parent"
|
||||||
android:layout_weight="1" />
|
android:layout_weight="1" />
|
||||||
|
@ -12,15 +12,4 @@
|
|||||||
android:layout_width="fill_parent"
|
android:layout_width="fill_parent"
|
||||||
android:layout_height="fill_parent"/>
|
android:layout_height="fill_parent"/>
|
||||||
|
|
||||||
<org.mozilla.gecko.home.HomeBanner android:id="@+id/home_banner"
|
|
||||||
style="@style/Widget.HomeBanner"
|
|
||||||
android:layout_width="fill_parent"
|
|
||||||
android:layout_height="@dimen/home_banner_height"
|
|
||||||
android:background="@drawable/home_banner"
|
|
||||||
android:layout_gravity="bottom"
|
|
||||||
android:gravity="center_vertical"
|
|
||||||
android:visibility="gone"
|
|
||||||
android:clickable="true"
|
|
||||||
android:focusable="true"/>
|
|
||||||
|
|
||||||
</FrameLayout>
|
</FrameLayout>
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
android:layout_height="fill_parent"
|
android:layout_height="fill_parent"
|
||||||
android:orientation="vertical">
|
android:orientation="vertical">
|
||||||
|
|
||||||
<FrameLayout android:id="@+id/visited_page_container"
|
<FrameLayout android:id="@+id/history_page_container"
|
||||||
android:layout_width="fill_parent"
|
android:layout_width="fill_parent"
|
||||||
android:layout_height="0dp"
|
android:layout_height="0dp"
|
||||||
android:layout_weight="1" />
|
android:layout_weight="1" />
|
||||||
|
@ -1,13 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<!-- This Source Code Form is subject to the terms of the Mozilla Public
|
|
||||||
- License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
||||||
- file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
|
|
||||||
|
|
||||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
|
||||||
android:layout_width="fill_parent"
|
|
||||||
android:layout_height="fill_parent"
|
|
||||||
android:orientation="vertical">
|
|
||||||
|
|
||||||
<include layout="@layout/home_history_list"/>
|
|
||||||
|
|
||||||
</LinearLayout>
|
|
34
mobile/android/base/resources/layout/home_top_sites_page.xml
Normal file
34
mobile/android/base/resources/layout/home_top_sites_page.xml
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
- License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
- file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
|
||||||
|
|
||||||
|
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:layout_width="fill_parent"
|
||||||
|
android:layout_height="fill_parent"
|
||||||
|
android:orientation="vertical">
|
||||||
|
|
||||||
|
<ViewStub android:id="@+id/home_empty_view_stub"
|
||||||
|
android:layout="@layout/home_empty_page"
|
||||||
|
android:layout_width="fill_parent"
|
||||||
|
android:layout_height="fill_parent"/>
|
||||||
|
|
||||||
|
<org.mozilla.gecko.home.HomeListView
|
||||||
|
android:id="@+id/list"
|
||||||
|
style="@style/Widget.TopSitesListView"
|
||||||
|
android:layout_width="fill_parent"
|
||||||
|
android:layout_height="0dp"
|
||||||
|
android:layout_weight="1"/>
|
||||||
|
|
||||||
|
<org.mozilla.gecko.home.HomeBanner android:id="@+id/home_banner"
|
||||||
|
style="@style/Widget.HomeBanner"
|
||||||
|
android:layout_width="fill_parent"
|
||||||
|
android:layout_height="@dimen/home_banner_height"
|
||||||
|
android:background="@drawable/home_banner"
|
||||||
|
android:layout_gravity="bottom"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:visibility="gone"
|
||||||
|
android:clickable="true"
|
||||||
|
android:focusable="true"/>
|
||||||
|
|
||||||
|
</LinearLayout>
|
@ -62,9 +62,11 @@
|
|||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style name="Widget.BookmarksListView" parent="Widget.HomeListView">
|
<style name="Widget.BookmarksListView" parent="Widget.HomeListView">
|
||||||
|
<item name="android:paddingTop">30dp</item>
|
||||||
<item name="android:paddingLeft">32dp</item>
|
<item name="android:paddingLeft">32dp</item>
|
||||||
<item name="android:paddingRight">32dp</item>
|
<item name="android:paddingRight">32dp</item>
|
||||||
<item name="android:scrollbarStyle">outsideOverlay</item>
|
<item name="android:scrollbarStyle">outsideOverlay</item>
|
||||||
|
<item name="topDivider">true</item>
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style name="Widget.TopBookmarksView" parent="Widget.GridView">
|
<style name="Widget.TopBookmarksView" parent="Widget.GridView">
|
||||||
@ -76,10 +78,9 @@
|
|||||||
<item name="android:verticalSpacing">10dp</item>
|
<item name="android:verticalSpacing">10dp</item>
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style name="Widget.ReadingListView" parent="Widget.BookmarksListView">
|
<style name="Widget.TopSitesListView" parent="Widget.BookmarksListView"/>
|
||||||
<item name="android:paddingTop">30dp</item>
|
|
||||||
<item name="topDivider">true</item>
|
<style name="Widget.ReadingListView" parent="Widget.BookmarksListView"/>
|
||||||
</style>
|
|
||||||
|
|
||||||
<style name="Widget.HomeBanner">
|
<style name="Widget.HomeBanner">
|
||||||
<item name="android:paddingLeft">32dp</item>
|
<item name="android:paddingLeft">32dp</item>
|
||||||
|
@ -82,8 +82,8 @@
|
|||||||
<!-- We need to maintain height for the tab widget on History Page
|
<!-- We need to maintain height for the tab widget on History Page
|
||||||
since android does not add footer/header divider height to its
|
since android does not add footer/header divider height to its
|
||||||
calculation for wrap_content in LinearLayout.
|
calculation for wrap_content in LinearLayout.
|
||||||
50dp * 3 Views + 30dp padding + 4dp dividers-->
|
50dp * 2 Views + 30dp padding + 4dp dividers-->
|
||||||
<dimen name="history_tab_widget_height">184dp</dimen>
|
<dimen name="history_tab_widget_height">134dp</dimen>
|
||||||
|
|
||||||
<!-- PageActionButtons dimensions -->
|
<!-- PageActionButtons dimensions -->
|
||||||
<dimen name="page_action_button_width">32dp</dimen>
|
<dimen name="page_action_button_width">32dp</dimen>
|
||||||
|
@ -158,6 +158,8 @@
|
|||||||
<item name="android:divider">#E7ECF0</item>
|
<item name="android:divider">#E7ECF0</item>
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
|
<style name="Widget.TopSitesListView" parent="Widget.BookmarksListView"/>
|
||||||
|
|
||||||
<style name="Widget.ReadingListView" parent="Widget.BookmarksListView"/>
|
<style name="Widget.ReadingListView" parent="Widget.BookmarksListView"/>
|
||||||
|
|
||||||
<style name="Widget.HomeBanner"/>
|
<style name="Widget.HomeBanner"/>
|
||||||
|
@ -252,6 +252,7 @@
|
|||||||
<string name="button_yes">&button_yes;</string>
|
<string name="button_yes">&button_yes;</string>
|
||||||
<string name="button_no">&button_no;</string>
|
<string name="button_no">&button_no;</string>
|
||||||
|
|
||||||
|
<string name="home_top_sites_title">&home_top_sites_title;</string>
|
||||||
<string name="home_history_title">&home_history_title;</string>
|
<string name="home_history_title">&home_history_title;</string>
|
||||||
<string name="home_last_tabs_title">&home_last_tabs_title;</string>
|
<string name="home_last_tabs_title">&home_last_tabs_title;</string>
|
||||||
<string name="home_last_tabs_open">&home_last_tabs_open;</string>
|
<string name="home_last_tabs_open">&home_last_tabs_open;</string>
|
||||||
|
Loading…
Reference in New Issue
Block a user