mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-06-04 15:38:59 +00:00
421 lines
16 KiB
Java
421 lines
16 KiB
Java
/* -*- Mode: Java; c-basic-offset: 4; tab-width: 20; indent-tabs-mode: nil; -*-
|
|
* 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/. */
|
|
|
|
package org.mozilla.goanna.home;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
|
|
import org.json.JSONArray;
|
|
import org.json.JSONException;
|
|
import org.json.JSONObject;
|
|
import org.mozilla.goanna.AboutPages;
|
|
import org.mozilla.goanna.EventDispatcher;
|
|
import org.mozilla.goanna.GoannaAppShell;
|
|
import org.mozilla.goanna.GoannaEvent;
|
|
import org.mozilla.goanna.GoannaProfile;
|
|
import org.mozilla.goanna.R;
|
|
import org.mozilla.goanna.SessionParser;
|
|
import org.mozilla.goanna.Telemetry;
|
|
import org.mozilla.goanna.TelemetryContract;
|
|
import org.mozilla.goanna.db.BrowserContract.CommonColumns;
|
|
import org.mozilla.goanna.db.BrowserContract.URLColumns;
|
|
import org.mozilla.goanna.util.EventCallback;
|
|
import org.mozilla.goanna.util.NativeEventListener;
|
|
import org.mozilla.goanna.util.NativeJSObject;
|
|
import org.mozilla.goanna.util.ThreadUtils;
|
|
|
|
import android.content.Context;
|
|
import android.database.Cursor;
|
|
import android.database.MatrixCursor;
|
|
import android.database.MatrixCursor.RowBuilder;
|
|
import android.os.Bundle;
|
|
import android.support.v4.content.Loader;
|
|
import android.util.Log;
|
|
import android.view.LayoutInflater;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
import android.view.ViewStub;
|
|
import android.widget.AdapterView;
|
|
import android.widget.ImageView;
|
|
import android.widget.TextView;
|
|
|
|
/**
|
|
* Fragment that displays tabs from last session in a ListView.
|
|
*/
|
|
public class RecentTabsPanel extends HomeFragment
|
|
implements NativeEventListener {
|
|
// Logging tag name
|
|
@SuppressWarnings("unused")
|
|
private static final String LOGTAG = "GoannaRecentTabsPanel";
|
|
|
|
// Cursor loader ID for the loader that loads recent tabs
|
|
private static final int LOADER_ID_RECENT_TABS = 0;
|
|
|
|
// Adapter for the list of recent tabs.
|
|
private RecentTabsAdapter mAdapter;
|
|
|
|
// The view shown by the fragment.
|
|
private HomeListView mList;
|
|
|
|
// Reference to the View to display when there are no results.
|
|
private View mEmptyView;
|
|
|
|
// Callbacks used for the search and favicon cursor loaders
|
|
private CursorLoaderCallbacks mCursorLoaderCallbacks;
|
|
|
|
// Recently closed tabs from goanna
|
|
private ClosedTab[] mClosedTabs;
|
|
|
|
private void restoreSessionWithHistory(List<String> dataList) {
|
|
JSONObject json = new JSONObject();
|
|
try {
|
|
json.put("tabs", new JSONArray(dataList));
|
|
} catch (JSONException e) {
|
|
Log.e(LOGTAG, "JSON error", e);
|
|
}
|
|
|
|
GoannaAppShell.sendEventToGoanna(GoannaEvent.createBroadcastEvent("Session:RestoreRecentTabs", json.toString()));
|
|
}
|
|
|
|
private static final class ClosedTab {
|
|
public final String url;
|
|
public final String title;
|
|
public final String data;
|
|
|
|
public ClosedTab(String url, String title, String data) {
|
|
this.url = url;
|
|
this.title = title;
|
|
this.data = data;
|
|
}
|
|
}
|
|
|
|
public static final class RecentTabs implements URLColumns, CommonColumns {
|
|
public static final String TYPE = "type";
|
|
public static final String DATA = "data";
|
|
|
|
public static final int TYPE_HEADER = 0;
|
|
public static final int TYPE_LAST_TIME = 1;
|
|
public static final int TYPE_CLOSED = 2;
|
|
public static final int TYPE_OPEN_ALL_LAST_TIME = 3;
|
|
public static final int TYPE_OPEN_ALL_CLOSED = 4;
|
|
}
|
|
|
|
@Override
|
|
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
|
return inflater.inflate(R.layout.home_recent_tabs_panel, container, false);
|
|
}
|
|
|
|
@Override
|
|
public void onViewCreated(View view, Bundle savedInstanceState) {
|
|
mList = (HomeListView) view.findViewById(R.id.list);
|
|
mList.setTag(HomePager.LIST_TAG_RECENT_TABS);
|
|
|
|
mList.setOnItemClickListener(new AdapterView.OnItemClickListener() {
|
|
@Override
|
|
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
|
|
final Cursor c = mAdapter.getCursor();
|
|
if (c == null || !c.moveToPosition(position)) {
|
|
return;
|
|
}
|
|
|
|
final int itemType = c.getInt(c.getColumnIndexOrThrow(RecentTabs.TYPE));
|
|
|
|
if (itemType == RecentTabs.TYPE_OPEN_ALL_LAST_TIME) {
|
|
openTabsWithType(RecentTabs.TYPE_LAST_TIME);
|
|
return;
|
|
}
|
|
|
|
if (itemType == RecentTabs.TYPE_OPEN_ALL_CLOSED) {
|
|
openTabsWithType(RecentTabs.TYPE_CLOSED);
|
|
return;
|
|
}
|
|
|
|
Telemetry.sendUIEvent(TelemetryContract.Event.LOAD_URL, TelemetryContract.Method.LIST_ITEM);
|
|
|
|
final List<String> dataList = new ArrayList<>();
|
|
dataList.add(c.getString(c.getColumnIndexOrThrow(RecentTabs.DATA)));
|
|
restoreSessionWithHistory(dataList);
|
|
}
|
|
});
|
|
|
|
mList.setContextMenuInfoFactory(new HomeContextMenuInfo.Factory() {
|
|
@Override
|
|
public HomeContextMenuInfo makeInfoForCursor(View view, int position, long id, Cursor cursor) {
|
|
// Don't show context menus for the "Open all" rows.
|
|
final int itemType = cursor.getInt(cursor.getColumnIndexOrThrow(RecentTabs.TYPE));
|
|
if (itemType == RecentTabs.TYPE_OPEN_ALL_LAST_TIME || itemType == RecentTabs.TYPE_OPEN_ALL_CLOSED) {
|
|
return null;
|
|
}
|
|
|
|
final HomeContextMenuInfo info = new HomeContextMenuInfo(view, position, id);
|
|
info.url = cursor.getString(cursor.getColumnIndexOrThrow(RecentTabs.URL));
|
|
info.title = cursor.getString(cursor.getColumnIndexOrThrow(RecentTabs.TITLE));
|
|
return info;
|
|
}
|
|
});
|
|
|
|
registerForContextMenu(mList);
|
|
|
|
EventDispatcher.getInstance().registerGoannaThreadListener(this, "ClosedTabs:Data");
|
|
GoannaAppShell.sendEventToGoanna(GoannaEvent.createBroadcastEvent("ClosedTabs:StartNotifications", null));
|
|
}
|
|
|
|
@Override
|
|
public void onDestroyView() {
|
|
super.onDestroyView();
|
|
mList = null;
|
|
mEmptyView = null;
|
|
|
|
EventDispatcher.getInstance().unregisterGoannaThreadListener(this, "ClosedTabs:Data");
|
|
GoannaAppShell.sendEventToGoanna(GoannaEvent.createBroadcastEvent("ClosedTabs:StopNotifications", null));
|
|
}
|
|
|
|
@Override
|
|
public void onActivityCreated(Bundle savedInstanceState) {
|
|
super.onActivityCreated(savedInstanceState);
|
|
|
|
// Intialize adapter
|
|
mAdapter = new RecentTabsAdapter(getActivity());
|
|
mList.setAdapter(mAdapter);
|
|
|
|
// Create callbacks before the initial loader is started
|
|
mCursorLoaderCallbacks = new CursorLoaderCallbacks();
|
|
loadIfVisible();
|
|
}
|
|
|
|
private void updateUiFromCursor(Cursor c) {
|
|
if (c != null && c.getCount() > 0) {
|
|
return;
|
|
}
|
|
|
|
if (mEmptyView == null) {
|
|
// Set empty panel view. We delay this so that the empty view won't flash.
|
|
final ViewStub emptyViewStub = (ViewStub) getView().findViewById(R.id.home_empty_view_stub);
|
|
mEmptyView = emptyViewStub.inflate();
|
|
|
|
final ImageView emptyIcon = (ImageView) mEmptyView.findViewById(R.id.home_empty_image);
|
|
emptyIcon.setImageResource(R.drawable.icon_last_tabs_empty);
|
|
|
|
final TextView emptyText = (TextView) mEmptyView.findViewById(R.id.home_empty_text);
|
|
emptyText.setText(R.string.home_last_tabs_empty);
|
|
|
|
mList.setEmptyView(mEmptyView);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void load() {
|
|
getLoaderManager().initLoader(LOADER_ID_RECENT_TABS, null, mCursorLoaderCallbacks);
|
|
}
|
|
|
|
@Override
|
|
public void handleMessage(String event, NativeJSObject message, EventCallback callback) {
|
|
final NativeJSObject[] tabs = message.getObjectArray("tabs");
|
|
final int length = tabs.length;
|
|
|
|
final ClosedTab[] closedTabs = new ClosedTab[length];
|
|
for (int i = 0; i < length; i++) {
|
|
final NativeJSObject tab = tabs[i];
|
|
closedTabs[i] = new ClosedTab(tab.getString("url"), tab.getString("title"), tab.getObject("data").toString());
|
|
}
|
|
|
|
// Only modify mClosedTabs on the UI thread
|
|
ThreadUtils.postToUiThread(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
mClosedTabs = closedTabs;
|
|
|
|
// The fragment might have been detached before this code
|
|
// runs in the UI thread.
|
|
if (getActivity() != null) {
|
|
// Reload the cursor to show recently closed tabs.
|
|
getLoaderManager().restartLoader(LOADER_ID_RECENT_TABS, null, mCursorLoaderCallbacks);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
private void openTabsWithType(int type) {
|
|
final Cursor c = mAdapter.getCursor();
|
|
if (c == null || !c.moveToFirst()) {
|
|
return;
|
|
}
|
|
|
|
final List<String> dataList = new ArrayList<String>();
|
|
do {
|
|
if (c.getInt(c.getColumnIndexOrThrow(RecentTabs.TYPE)) == type) {
|
|
dataList.add(c.getString(c.getColumnIndexOrThrow(RecentTabs.DATA)));
|
|
}
|
|
} while (c.moveToNext());
|
|
|
|
Telemetry.sendUIEvent(TelemetryContract.Event.LOAD_URL, TelemetryContract.Method.BUTTON);
|
|
|
|
restoreSessionWithHistory(dataList);
|
|
}
|
|
|
|
private static class RecentTabsCursorLoader extends SimpleCursorLoader {
|
|
private final ClosedTab[] closedTabs;
|
|
|
|
public RecentTabsCursorLoader(Context context, ClosedTab[] closedTabs) {
|
|
super(context);
|
|
this.closedTabs = closedTabs;
|
|
}
|
|
|
|
private void addRow(MatrixCursor c, String url, String title, int type, String data) {
|
|
final RowBuilder row = c.newRow();
|
|
row.add(-1);
|
|
row.add(url);
|
|
row.add(title);
|
|
row.add(type);
|
|
row.add(data);
|
|
}
|
|
|
|
@Override
|
|
public Cursor loadCursor() {
|
|
final Context context = getContext();
|
|
|
|
final MatrixCursor c = new MatrixCursor(new String[] { RecentTabs._ID,
|
|
RecentTabs.URL,
|
|
RecentTabs.TITLE,
|
|
RecentTabs.TYPE,
|
|
RecentTabs.DATA});
|
|
|
|
if (closedTabs != null && closedTabs.length > 0) {
|
|
// How many closed tabs are actually displayed.
|
|
int visibleClosedTabs = 0;
|
|
|
|
final int length = closedTabs.length;
|
|
for (int i = 0; i < length; i++) {
|
|
final String url = closedTabs[i].url;
|
|
|
|
// Don't show recent tabs for about:home or about:privatebrowsing.
|
|
if (!AboutPages.isTitlelessAboutPage(url)) {
|
|
// If this is the first closed tab we're adding, add a header for the section.
|
|
if (visibleClosedTabs == 0) {
|
|
addRow(c, null, context.getString(R.string.home_closed_tabs_title), RecentTabs.TYPE_HEADER, null);
|
|
}
|
|
addRow(c, url, closedTabs[i].title, RecentTabs.TYPE_CLOSED, closedTabs[i].data);
|
|
visibleClosedTabs++;
|
|
}
|
|
}
|
|
|
|
// Add an "Open all" button if more than 2 tabs were added to the list.
|
|
if (visibleClosedTabs > 1) {
|
|
addRow(c, null, null, RecentTabs.TYPE_OPEN_ALL_CLOSED, null);
|
|
}
|
|
}
|
|
|
|
final String jsonString = GoannaProfile.get(context).readSessionFile(true);
|
|
if (jsonString == null) {
|
|
// No previous session data
|
|
return c;
|
|
}
|
|
|
|
final int count = c.getCount();
|
|
|
|
new SessionParser() {
|
|
@Override
|
|
public void onTabRead(SessionTab tab) {
|
|
final String url = tab.getUrl();
|
|
|
|
// Don't show last tabs for about:home
|
|
if (AboutPages.isAboutHome(url)) {
|
|
return;
|
|
}
|
|
|
|
// If this is the first tab we're reading, add a header.
|
|
if (c.getCount() == count) {
|
|
addRow(c, null, context.getString(R.string.home_last_tabs_title), RecentTabs.TYPE_HEADER, null);
|
|
}
|
|
|
|
addRow(c, url, tab.getTitle(), RecentTabs.TYPE_LAST_TIME, tab.getTabObject().toString());
|
|
}
|
|
}.parse(jsonString);
|
|
|
|
// Add an "Open all" button if more than 2 tabs were added to the list (account for the header)
|
|
if (c.getCount() - count > 2) {
|
|
addRow(c, null, null, RecentTabs.TYPE_OPEN_ALL_LAST_TIME, null);
|
|
}
|
|
|
|
return c;
|
|
}
|
|
}
|
|
|
|
private static class RecentTabsAdapter extends MultiTypeCursorAdapter {
|
|
private static final int ROW_HEADER = 0;
|
|
private static final int ROW_STANDARD = 1;
|
|
private static final int ROW_OPEN_ALL = 2;
|
|
|
|
private static final int[] VIEW_TYPES = new int[] { ROW_STANDARD, ROW_HEADER, ROW_OPEN_ALL };
|
|
private static final int[] LAYOUT_TYPES =
|
|
new int[] { R.layout.home_item_row, R.layout.home_header_row, R.layout.home_open_all_row };
|
|
|
|
public RecentTabsAdapter(Context context) {
|
|
super(context, null, VIEW_TYPES, LAYOUT_TYPES);
|
|
}
|
|
|
|
@Override
|
|
public int getItemViewType(int position) {
|
|
final Cursor c = getCursor(position);
|
|
final int type = c.getInt(c.getColumnIndexOrThrow(RecentTabs.TYPE));
|
|
|
|
if (type == RecentTabs.TYPE_HEADER) {
|
|
return ROW_HEADER;
|
|
}
|
|
|
|
if (type == RecentTabs.TYPE_OPEN_ALL_LAST_TIME || type == RecentTabs.TYPE_OPEN_ALL_CLOSED) {
|
|
return ROW_OPEN_ALL;
|
|
}
|
|
|
|
return ROW_STANDARD;
|
|
}
|
|
|
|
@Override
|
|
public boolean isEnabled(int position) {
|
|
return (getItemViewType(position) != ROW_HEADER);
|
|
}
|
|
|
|
@Override
|
|
public void bindView(View view, Context context, int position) {
|
|
final int itemType = getItemViewType(position);
|
|
if (itemType == ROW_OPEN_ALL) {
|
|
return;
|
|
}
|
|
|
|
final Cursor c = getCursor(position);
|
|
|
|
if (itemType == ROW_HEADER) {
|
|
final String title = c.getString(c.getColumnIndexOrThrow(RecentTabs.TITLE));
|
|
final TextView textView = (TextView) view;
|
|
textView.setText(title);
|
|
} else if (itemType == ROW_STANDARD) {
|
|
final TwoLinePageRow pageRow = (TwoLinePageRow) view;
|
|
pageRow.setShowIcons(false);
|
|
pageRow.updateFromCursor(c);
|
|
}
|
|
}
|
|
}
|
|
|
|
private class CursorLoaderCallbacks extends TransitionAwareCursorLoaderCallbacks {
|
|
@Override
|
|
public Loader<Cursor> onCreateLoader(int id, Bundle args) {
|
|
return new RecentTabsCursorLoader(getActivity(), mClosedTabs);
|
|
}
|
|
|
|
@Override
|
|
public void onLoadFinishedAfterTransitions(Loader<Cursor> loader, Cursor c) {
|
|
mAdapter.swapCursor(c);
|
|
updateUiFromCursor(c);
|
|
}
|
|
|
|
@Override
|
|
public void onLoaderReset(Loader<Cursor> loader) {
|
|
super.onLoaderReset(loader);
|
|
mAdapter.swapCursor(null);
|
|
}
|
|
}
|
|
}
|