/* * Copyright (C) 2012 Google Inc. * Licensed to The Android Open Source Project. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.mail.ui; import android.app.Activity; import android.app.Fragment; import android.app.LoaderManager; import android.content.Context; import android.content.Loader; import android.database.Cursor; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.support.annotation.Nullable; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import com.android.emailcommon.mail.Address; import com.android.mail.R; import com.android.mail.analytics.Analytics; import com.android.mail.browse.ConversationAccountController; import com.android.mail.browse.ConversationMessage; import com.android.mail.browse.ConversationViewHeader.ConversationViewHeaderCallbacks; import com.android.mail.browse.MessageCursor; import com.android.mail.browse.MessageCursor.ConversationController; import com.android.mail.content.ObjectCursor; import com.android.mail.content.ObjectCursorLoader; import com.android.mail.providers.Account; import com.android.mail.providers.AccountObserver; import com.android.mail.providers.Conversation; import com.android.mail.providers.Folder; import com.android.mail.providers.ListParams; import com.android.mail.providers.Settings; import com.android.mail.providers.UIProvider; import com.android.mail.providers.UIProvider.CursorStatus; import com.android.mail.utils.LogTag; import com.android.mail.utils.LogUtils; import com.android.mail.utils.Utils; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; import java.util.Map; public abstract class AbstractConversationViewFragment extends Fragment implements ConversationController, ConversationAccountController, ConversationViewHeaderCallbacks { protected static final String ARG_ACCOUNT = "account"; public static final String ARG_CONVERSATION = "conversation"; private static final String LOG_TAG = LogTag.getLogTag(); protected static final int MESSAGE_LOADER = 0; protected static final int CONTACT_LOADER = 1; public static final int ATTACHMENT_OPTION1_LOADER = 2; protected ControllableActivity mActivity; private final MessageLoaderCallbacks mMessageLoaderCallbacks = new MessageLoaderCallbacks(); private ContactLoaderCallbacks mContactLoaderCallbacks; private MenuItem mChangeFoldersMenuItem; protected Conversation mConversation; protected String mBaseUri; protected Account mAccount; /** * Must be instantiated in a derived class's onCreate. */ protected AbstractConversationWebViewClient mWebViewClient; /** * Cache of email address strings to parsed Address objects. *

* Remember to synchronize on the map when reading or writing to this cache, because some * instances use it off the UI thread (e.g. from WebView). */ protected final Map mAddressCache = Collections.synchronizedMap( new HashMap()); private MessageCursor mCursor; private Context mContext; /** * A backwards-compatible version of {{@link #getUserVisibleHint()}. Like the framework flag, * this flag is saved and restored. */ private boolean mUserVisible; private final Handler mHandler = new Handler(); /** True if we want to avoid marking the conversation as viewed and read. */ private boolean mSuppressMarkingViewed; /** * Parcelable state of the conversation view. Can safely be used without null checking any time * after {@link #onCreate(Bundle)}. */ protected ConversationViewState mViewState; private boolean mIsDetached; private boolean mHasConversationBeenTransformed; private boolean mHasConversationTransformBeenReverted; protected boolean mConversationSeen = false; private final AccountObserver mAccountObserver = new AccountObserver() { @Override public void onChanged(Account newAccount) { final Account oldAccount = mAccount; mAccount = newAccount; mWebViewClient.setAccount(mAccount); onAccountChanged(newAccount, oldAccount); } }; private static final String BUNDLE_VIEW_STATE = AbstractConversationViewFragment.class.getName() + "viewstate"; /** * We save the user visible flag so the various transitions that occur during rotation do not * cause unnecessary visibility change. */ private static final String BUNDLE_USER_VISIBLE = AbstractConversationViewFragment.class.getName() + "uservisible"; private static final String BUNDLE_DETACHED = AbstractConversationViewFragment.class.getName() + "detached"; private static final String BUNDLE_KEY_HAS_CONVERSATION_BEEN_TRANSFORMED = AbstractConversationViewFragment.class.getName() + "conversationtransformed"; private static final String BUNDLE_KEY_HAS_CONVERSATION_BEEN_REVERTED = AbstractConversationViewFragment.class.getName() + "conversationreverted"; public static Bundle makeBasicArgs(Account account) { Bundle args = new Bundle(); args.putParcelable(ARG_ACCOUNT, account); return args; } /** * Constructor needs to be public to handle orientation changes and activity * lifecycle events. */ public AbstractConversationViewFragment() { super(); } /** * Subclasses must override, since this depends on how many messages are * shown in the conversation view. */ protected void markUnread() { // Do not automatically mark this conversation viewed and read. mSuppressMarkingViewed = true; } /** * Marks a conversation either 'seen' (force=false), as in when the conversation is made visible * and should be marked read, or 'read' (force=true), as in when the action bar menu item to * mark this conversation read is selected. * * @param force true to force marking it read, false to allow peek mode to prevent it */ private final void markRead(boolean force) { final ControllableActivity activity = (ControllableActivity) getActivity(); if (activity == null) { return; } // mark viewed/read if not previously marked viewed by this conversation view, // or if unread messages still exist in the message list cursor // we don't want to keep marking viewed on rotation or restore // but we do want future re-renders to mark read (e.g. "New message from X" case) final MessageCursor cursor = getMessageCursor(); LogUtils.d(LOG_TAG, "onConversationSeen() - mConversation.isViewed() = %b, " + "cursor null = %b, cursor.isConversationRead() = %b", mConversation.isViewed(), cursor == null, cursor != null && cursor.isConversationRead()); if (!mConversation.isViewed() || (cursor != null && !cursor.isConversationRead())) { // Mark the conversation read no matter what if force=true. // else only mark it seen if appropriate (2-pane peek=true doesn't mark things seen) final boolean convMarkedRead; if (force) { activity.getConversationUpdater() .markConversationsRead(Arrays.asList(mConversation), true /* read */, true /* viewed */); convMarkedRead = true; } else { convMarkedRead = activity.getConversationUpdater() .markConversationSeen(mConversation); } // and update the Message objects in the cursor so the next time a cursor update // happens with these messages marked read, we know to ignore it if (convMarkedRead && cursor != null && !cursor.isClosed()) { cursor.markMessagesRead(); } } } /** * Subclasses must override this, since they may want to display a single or * many messages related to this conversation. */ protected abstract void onMessageCursorLoadFinished( Loader> loader, MessageCursor newCursor, MessageCursor oldCursor); /** * Subclasses must override this, since they may want to display a single or * many messages related to this conversation. */ @Override public abstract void onConversationViewHeaderHeightChange(int newHeight); public abstract void onUserVisibleHintChanged(); /** * Subclasses must override this. */ protected abstract void onAccountChanged(Account newAccount, Account oldAccount); @Override public void onCreate(Bundle savedState) { super.onCreate(savedState); parseArguments(); setBaseUri(); LogUtils.d(LOG_TAG, "onCreate in ConversationViewFragment (this=%s)", this); // Not really, we just want to get a crack to store a reference to the change_folder item setHasOptionsMenu(true); if (savedState != null) { mViewState = savedState.getParcelable(BUNDLE_VIEW_STATE); mUserVisible = savedState.getBoolean(BUNDLE_USER_VISIBLE); mIsDetached = savedState.getBoolean(BUNDLE_DETACHED, false); mHasConversationBeenTransformed = savedState.getBoolean(BUNDLE_KEY_HAS_CONVERSATION_BEEN_TRANSFORMED, false); mHasConversationTransformBeenReverted = savedState.getBoolean(BUNDLE_KEY_HAS_CONVERSATION_BEEN_REVERTED, false); } else { mViewState = getNewViewState(); mHasConversationBeenTransformed = false; mHasConversationTransformBeenReverted = false; } } /** * Can be overridden in case a subclass needs to get additional arguments. */ protected void parseArguments() { final Bundle args = getArguments(); mAccount = args.getParcelable(ARG_ACCOUNT); mConversation = args.getParcelable(ARG_CONVERSATION); } /** * Can be overridden in case a subclass needs a different uri format * (such as one that does not rely on account and/or conversation. */ protected void setBaseUri() { mBaseUri = buildBaseUri(getContext(), mAccount, mConversation); } public static String buildBaseUri(Context context, Account account, Conversation conversation) { // Since the uri specified in the conversation base uri may not be unique, we specify a // base uri that us guaranteed to be unique for this conversation. return "x-thread://" + account.getAccountId().hashCode() + "/" + conversation.id; } @Override public String toString() { // log extra info at DEBUG level or finer final String s = super.toString(); if (!LogUtils.isLoggable(LOG_TAG, LogUtils.DEBUG) || mConversation == null) { return s; } return "(" + s + " conv=" + mConversation + ")"; } @Override public void onActivityCreated(Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); final Activity activity = getActivity(); if (!(activity instanceof ControllableActivity)) { LogUtils.wtf(LOG_TAG, "ConversationViewFragment expects only a ControllableActivity to" + "create it. Cannot proceed."); } if (activity == null || activity.isFinishing()) { // Activity is finishing, just bail. return; } mActivity = (ControllableActivity) activity; mContext = activity.getApplicationContext(); mWebViewClient.setActivity(activity); mAccount = mAccountObserver.initialize(mActivity.getAccountController()); mWebViewClient.setAccount(mAccount); } @Override public ConversationUpdater getListController() { final ControllableActivity activity = (ControllableActivity) getActivity(); return activity != null ? activity.getConversationUpdater() : null; } public Context getContext() { return mContext; } @Override public Conversation getConversation() { return mConversation; } @Override public @Nullable MessageCursor getMessageCursor() { return mCursor; } public Handler getHandler() { return mHandler; } public MessageLoaderCallbacks getMessageLoaderCallbacks() { return mMessageLoaderCallbacks; } public ContactLoaderCallbacks getContactInfoSource() { if (mContactLoaderCallbacks == null) { mContactLoaderCallbacks = mActivity.getContactLoaderCallbacks(); } return mContactLoaderCallbacks; } @Override public Account getAccount() { return mAccount; } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { super.onCreateOptionsMenu(menu, inflater); mChangeFoldersMenuItem = menu.findItem(R.id.change_folders); } @Override public boolean onOptionsItemSelected(MenuItem item) { if (!isUserVisible()) { // Unclear how this is happening. Current theory is that this fragment was scheduled // to be removed, but the remove transaction failed. When the Activity is later // restored, the FragmentManager restores this fragment, but Fragment.mMenuVisible is // stuck at its initial value (true), which makes this zombie fragment eligible for // menu item clicks. // // Work around this by relying on the (properly restored) extra user visible hint. LogUtils.e(LOG_TAG, "ACVF ignoring onOptionsItemSelected b/c userVisibleHint is false. f=%s", this); if (LogUtils.isLoggable(LOG_TAG, LogUtils.DEBUG)) { LogUtils.e(LOG_TAG, "%s", Utils.dumpFragment(this)); } return false; } boolean handled = true; final int itemId = item.getItemId(); if (itemId == R.id.inside_conversation_unread || itemId == R.id.toggle_read_unread) { markUnread(); } else if (itemId == R.id.read) { markRead(true /* force */); mActivity.supportInvalidateOptionsMenu(); } else if (itemId == R.id.show_original) { showUntransformedConversation(); } else if (itemId == R.id.print_all) { printConversation(); } else if (itemId == R.id.reply) { handleReply(); } else if (itemId == R.id.reply_all) { handleReplyAll(); } else { handled = false; } return handled; } @Override public void onPrepareOptionsMenu(Menu menu) { // Only show option if we support message transforms and message has been transformed. Utils.setMenuItemPresent(menu, R.id.show_original, supportsMessageTransforms() && mHasConversationBeenTransformed && !mHasConversationTransformBeenReverted); final MenuItem printMenuItem = menu.findItem(R.id.print_all); if (printMenuItem != null) { // compute the visibility of the print menu item printMenuItem.setVisible(Utils.isRunningKitkatOrLater() && shouldShowPrintInOverflow()); // compute the text displayed on the print menu item if (mConversation.getNumMessages() == 1) { printMenuItem.setTitle(R.string.print); } else { printMenuItem.setTitle(R.string.print_all); } } } abstract boolean supportsMessageTransforms(); // BEGIN conversation header callbacks @Override public void onFoldersClicked() { if (mChangeFoldersMenuItem == null) { LogUtils.e(LOG_TAG, "unable to open 'change folders' dialog for a conversation"); return; } mActivity.onOptionsItemSelected(mChangeFoldersMenuItem); } // END conversation header callbacks @Override public void onStart() { super.onStart(); Analytics.getInstance().sendView(getClass().getName()); } @Override public void onSaveInstanceState(Bundle outState) { if (mViewState != null) { outState.putParcelable(BUNDLE_VIEW_STATE, mViewState); } outState.putBoolean(BUNDLE_USER_VISIBLE, mUserVisible); outState.putBoolean(BUNDLE_DETACHED, mIsDetached); outState.putBoolean(BUNDLE_KEY_HAS_CONVERSATION_BEEN_TRANSFORMED, mHasConversationBeenTransformed); outState.putBoolean(BUNDLE_KEY_HAS_CONVERSATION_BEEN_REVERTED, mHasConversationTransformBeenReverted); } @Override public void onDestroyView() { super.onDestroyView(); mAccountObserver.unregisterAndDestroy(); } /** * {@link #setUserVisibleHint(boolean)} only works on API >= 15, so implement our own for * reliability on older platforms. */ public void setExtraUserVisibleHint(boolean isVisibleToUser) { LogUtils.v(LOG_TAG, "in CVF.setHint, val=%s (%s)", isVisibleToUser, this); if (mUserVisible != isVisibleToUser) { mUserVisible = isVisibleToUser; MessageCursor cursor = getMessageCursor(); if (mUserVisible && (cursor != null && cursor.isLoaded() && cursor.getCount() == 0)) { // Pop back to conversation list and show error. onError(); return; } onUserVisibleHintChanged(); } } public boolean isUserVisible() { return mUserVisible; } protected void timerMark(String msg) { if (isUserVisible()) { Utils.sConvLoadTimer.mark(msg); } } private class MessageLoaderCallbacks implements LoaderManager.LoaderCallbacks> { @Override public Loader> onCreateLoader(int id, Bundle args) { return new MessageLoader(mActivity.getActivityContext(), mConversation.messageListUri); } @Override public void onLoadFinished(Loader> loader, ObjectCursor data) { // ignore truly duplicate results // this can happen when restoring after rotation if (mCursor == data) { return; } else { final MessageCursor messageCursor = (MessageCursor) data; // bind the cursor to this fragment so it can access to the current list controller messageCursor.setController(AbstractConversationViewFragment.this); if (LogUtils.isLoggable(LOG_TAG, LogUtils.DEBUG)) { LogUtils.d(LOG_TAG, "LOADED CONVERSATION= %s", messageCursor.getDebugDump()); } // We have no messages: exit conversation view. if (messageCursor.getCount() == 0 && (!CursorStatus.isWaitingForResults(messageCursor.getStatus()) || mIsDetached)) { if (mUserVisible) { onError(); } else { // we expect that the pager adapter will remove this // conversation fragment on its own due to a separate // conversation cursor update (we might get here if the // message list update fires first. nothing to do // because we expect to be torn down soon.) LogUtils.i(LOG_TAG, "CVF: offscreen conv has no messages, ignoring update" + " in anticipation of conv cursor update. c=%s", mConversation.uri); } // existing mCursor will imminently be closed, must stop referencing it // since we expect to be kicked out soon, it doesn't matter what mCursor // becomes mCursor = null; return; } // ignore cursors that are still loading results if (!messageCursor.isLoaded()) { // existing mCursor will imminently be closed, must stop referencing it // in this case, the new cursor is also no good, and since don't expect to get // here except in initial load situations, it's safest to just ensure the // reference is null mCursor = null; return; } final MessageCursor oldCursor = mCursor; mCursor = messageCursor; onMessageCursorLoadFinished(loader, mCursor, oldCursor); } } @Override public void onLoaderReset(Loader> loader) { mCursor = null; } } private void onError() { // need to exit this view- conversation may have been // deleted, or for whatever reason is now invalid (e.g. // discard single draft) // // N.B. this may involve a fragment transaction, which // FragmentManager will refuse to execute directly // within onLoadFinished. Make sure the controller knows. LogUtils.i(LOG_TAG, "CVF: visible conv has no messages, exiting conv mode"); // TODO(mindyp): handle ERROR status by showing an error // message to the user that there are no messages in // this conversation popOut(); } private void popOut() { mHandler.post(new FragmentRunnable("popOut", this) { @Override public void go() { if (mActivity != null) { mActivity.getListHandler() .onConversationSelected(null, true /* inLoaderCallbacks */); } } }); } /** * @see Folder#getTypeDescription() */ protected String getCurrentFolderTypeDesc() { final Folder currFolder; if (mActivity != null) { currFolder = mActivity.getFolderController().getFolder(); } else { currFolder = null; } final String folderStr; if (currFolder != null) { folderStr = currFolder.getTypeDescription(); } else { folderStr = "unknown_folder"; } return folderStr; } private void logConversationView() { final String folderStr = getCurrentFolderTypeDesc(); Analytics.getInstance().sendEvent("view_conversation", folderStr, mConversation.isRemote ? "unsynced" : "synced", mConversation.getNumMessages()); } protected final void onConversationSeen() { LogUtils.d(LOG_TAG, "AbstractConversationViewFragment#onConversationSeen()"); // Ignore unsafe calls made after a fragment is detached from an activity final ControllableActivity activity = (ControllableActivity) getActivity(); if (activity == null) { LogUtils.w(LOG_TAG, "ignoring onConversationSeen for conv=%s", mConversation.id); return; } // this method is called 2x on rotation; debounce this a bit so as not to // dramatically skew analytics data too much. Ideally, it should be called zero times // on rotation... if (!mConversationSeen) { logConversationView(); } mViewState.setInfoForConversation(mConversation); LogUtils.d(LOG_TAG, "onConversationSeen() - mSuppressMarkingViewed = %b", mSuppressMarkingViewed); // In most circumstances we want to mark the conversation as viewed and read, since the // user has read it. However, if the user has already marked the conversation unread, we // do not want a later mark-read operation to undo this. So we check this variable which // is set in #markUnread() which suppresses automatic mark-read. if (!mSuppressMarkingViewed) { markRead(false /* force */); } activity.getListHandler().onConversationSeen(); mConversationSeen = true; } protected ConversationViewState getNewViewState() { return new ConversationViewState(); } private static class MessageLoader extends ObjectCursorLoader { private boolean mDeliveredFirstResults = false; public MessageLoader(Context c, Uri messageListUri) { super(c, messageListUri, UIProvider.MESSAGE_PROJECTION, ConversationMessage.FACTORY); } @Override public void deliverResult(ObjectCursor result) { // We want to deliver these results, and then we want to make sure // that any subsequent // queries do not hit the network super.deliverResult(result); if (!mDeliveredFirstResults) { mDeliveredFirstResults = true; Uri uri = getUri(); // Create a ListParams that tells the provider to not hit the // network final ListParams listParams = new ListParams(ListParams.NO_LIMIT, false /* useNetwork */); // Build the new uri with this additional parameter uri = uri .buildUpon() .appendQueryParameter(UIProvider.LIST_PARAMS_QUERY_PARAMETER, listParams.serialize()).build(); setUri(uri); } } @Override protected ObjectCursor getObjectCursor(Cursor inner) { return new MessageCursor(inner); } } public abstract void onConversationUpdated(Conversation conversation); public void onDetachedModeEntered() { // If we have no messages, then we have nothing to display, so leave this view. // Otherwise, just set the detached flag. final Cursor messageCursor = getMessageCursor(); if (messageCursor == null || messageCursor.getCount() == 0) { popOut(); } else { mIsDetached = true; } } /** * Called when the JavaScript reports that it transformed a message. * Sets a flag to true and invalidates the options menu so it will * include the "Revert auto-sizing" menu option. */ public void onConversationTransformed() { mHasConversationBeenTransformed = true; mHandler.post(new FragmentRunnable("invalidateOptionsMenu", this) { @Override public void go() { mActivity.supportInvalidateOptionsMenu(); } }); } /** * Called when the "Revert auto-sizing" option is selected. Default * implementation simply sets a value on whether transforms should be * applied. Derived classes should override this class and force a * re-render so that the conversation renders without */ public void showUntransformedConversation() { // must set the value to true so we don't show the options menu item again mHasConversationTransformBeenReverted = true; } /** * Returns {@code true} if the conversation should be transformed. {@code false}, otherwise. * @return {@code true} if the conversation should be transformed. {@code false}, otherwise. */ public boolean shouldApplyTransforms() { return (mAccount.enableMessageTransforms > 0) && !mHasConversationTransformBeenReverted; } /** * The Print item in the overflow menu of the Conversation view is shown based on the return * from this method. * * @return {@code true} if the conversation can be printed; {@code false} otherwise. */ protected abstract boolean shouldShowPrintInOverflow(); /** * Prints all messages in the conversation. */ protected abstract void printConversation(); // These methods should perform default reply/replyall action on the last message. protected abstract void handleReply(); protected abstract void handleReplyAll(); public boolean shouldAlwaysShowImages() { return (mAccount != null) && (mAccount.settings.showImages == Settings.ShowImages.ALWAYS); } }