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:
Lucas Rocha 2013-09-17 16:58:12 -04:00
parent 356f594ec7
commit ed9279760c
18 changed files with 188 additions and 538 deletions

View File

@ -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();
} }

View File

@ -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 \

View File

@ -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;

View File

@ -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);
}
}
}
} }

View File

@ -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);

View File

@ -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) {

View File

@ -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);

View File

@ -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">

View File

@ -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" />

View File

@ -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" />

View File

@ -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>

View File

@ -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" />

View File

@ -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>

View 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>

View File

@ -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>

View File

@ -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>

View File

@ -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"/>

View File

@ -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>