blob: f0d191dd6a712e27fba8fbebe8cf337d4551f6de [file] [log] [blame]
/*
* 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.ActionBar;
import android.app.SearchManager;
import android.app.SearchableInfo;
import android.content.ContentResolver;
import android.content.Context;
import android.content.res.Resources;
import android.database.Cursor;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Build;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.support.v4.text.BidiFormatter;
import android.text.TextUtils;
import android.util.AttributeSet;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewConfiguration;
import android.widget.LinearLayout;
import android.widget.SearchView;
import android.widget.SearchView.OnQueryTextListener;
import android.widget.SearchView.OnSuggestionListener;
import android.widget.TextView;
import com.android.mail.ConversationListContext;
import com.android.mail.R;
import com.android.mail.preferences.MailPrefs;
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.FolderObserver;
import com.android.mail.providers.SearchRecentSuggestionsProvider;
import com.android.mail.providers.UIProvider;
import com.android.mail.providers.UIProvider.AccountCapabilities;
import com.android.mail.providers.UIProvider.FolderCapabilities;
import com.android.mail.providers.UIProvider.FolderType;
import com.android.mail.utils.LogTag;
import com.android.mail.utils.LogUtils;
import com.android.mail.utils.Utils;
/**
* View to manage the various states of the Mail Action Bar.
* <p>
* This also happens to be the custom view we supply to ActionBar.
*
*/
public class MailActionBarView extends LinearLayout implements ViewMode.ModeChangeListener,
OnQueryTextListener, OnSuggestionListener, MenuItem.OnActionExpandListener,
View.OnClickListener {
protected ActionBar mActionBar;
protected ControllableActivity mActivity;
protected ActivityController mController;
/**
* The current mode of the ActionBar and Activity
*/
private ViewMode mViewModeController;
private MenuItem mSearch;
/**
* The account currently being shown
*/
private Account mAccount;
/**
* The folder currently being shown
*/
private Folder mFolder;
private SearchView mSearchWidget;
private MenuItem mHelpItem;
private MenuItem mSendFeedbackItem;
private MenuItem mFolderSettingsItem;
private MenuItem mEmptyTrashItem;
private MenuItem mEmptySpamItem;
private boolean mUseLegacyTitle;
private View mLegacyTitleContainer;
private TextView mLegacyTitle;
private TextView mLegacySubTitle;
/** True if the current device is a tablet, false otherwise. */
protected final boolean mIsOnTablet;
private Conversation mCurrentConversation;
public static final String LOG_TAG = LogTag.getLogTag();
private FolderObserver mFolderObserver;
/** A handler that changes the subtitle when it receives a message. */
private final class SubtitleHandler extends Handler {
/** Message sent to display the account email address in the subtitle. */
private static final int EMAIL = 0;
@Override
public void handleMessage(Message message) {
assert (message.what == EMAIL);
final String subtitleText;
if (mAccount != null) {
// Display the account name (email address).
subtitleText = mAccount.name;
} else {
subtitleText = null;
LogUtils.wtf(LOG_TAG, "MABV.handleMessage() has a null account!");
}
setSubtitle(mBidiFormatter.unicodeWrap(subtitleText));
super.handleMessage(message);
}
}
/** Changes the subtitle to display the account name */
private final SubtitleHandler mHandler = new SubtitleHandler();
/** Unread count for the current folder. */
private int mUnreadCount = 0;
/** We show the email address after this delay: 5 seconds currently */
private static final int ACCOUNT_DELAY_MS = 5 * 1000;
/** At what point do we stop showing the unread count: 999+ currently */
private final int UNREAD_LIMIT;
/** BidiFormatter for title and subtitle. */
private final BidiFormatter mBidiFormatter;
/** Updates the resolver and tells it the most recent account. */
private final class UpdateProvider extends AsyncTask<Bundle, Void, Void> {
final Uri mAccount;
final ContentResolver mResolver;
public UpdateProvider(Uri account, ContentResolver resolver) {
mAccount = account;
mResolver = resolver;
}
@Override
protected Void doInBackground(Bundle... params) {
mResolver.call(mAccount, UIProvider.AccountCallMethods.SET_CURRENT_ACCOUNT,
mAccount.toString(), params[0]);
return null;
}
}
private final AccountObserver mAccountObserver = new AccountObserver() {
@Override
public void onChanged(Account newAccount) {
updateAccount(newAccount);
}
};
public MailActionBarView(Context context) {
this(context, null);
}
public MailActionBarView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public MailActionBarView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
final Resources r = getResources();
mIsOnTablet = Utils.useTabletUI(r);
UNREAD_LIMIT = r.getInteger(R.integer.maxUnreadCount);
mBidiFormatter = BidiFormatter.getInstance();
}
private void initializeTitleViews() {
mLegacyTitleContainer = findViewById(R.id.legacy_title_container);
if (mLegacyTitleContainer != null) {
// Determine if this device is running on MR1.1 or later
final boolean runningMR11OrLater = actionBarSupportsNewMethods(mActionBar);
if (runningMR11OrLater || !mController.isDrawerEnabled()) {
// We don't need the legacy view, just hide it
mLegacyTitleContainer.setVisibility(View.GONE);
mUseLegacyTitle = false;
} else {
mUseLegacyTitle = true;
// We need to show the legacy title/subtitle. Set the click listener
mLegacyTitleContainer.setOnClickListener(this);
mLegacyTitle = (TextView)mLegacyTitleContainer.findViewById(R.id.legacy_title);
mLegacySubTitle =
(TextView)mLegacyTitleContainer.findViewById(R.id.legacy_subtitle);
}
}
}
public void expandSearch() {
if (mSearch != null) {
mSearch.expandActionView();
}
}
/**
* Close the search view if it is expanded.
*/
public void collapseSearch() {
if (mSearch != null) {
mSearch.collapseActionView();
}
}
/**
* Get the search menu item.
*/
protected MenuItem getSearch() {
return mSearch;
}
public boolean onCreateOptionsMenu(Menu menu) {
// If the mode is valid, then set the initial menu
if (getMode() == ViewMode.UNKNOWN) {
return false;
}
mSearch = menu.findItem(R.id.search);
if (mSearch != null) {
mSearchWidget = (SearchView) mSearch.getActionView();
mSearch.setOnActionExpandListener(this);
SearchManager searchManager = (SearchManager) mActivity.getActivityContext()
.getSystemService(Context.SEARCH_SERVICE);
if (searchManager != null && mSearchWidget != null) {
SearchableInfo info = searchManager.getSearchableInfo(mActivity.getComponentName());
mSearchWidget.setSearchableInfo(info);
mSearchWidget.setOnQueryTextListener(this);
mSearchWidget.setOnSuggestionListener(this);
mSearchWidget.setIconifiedByDefault(true);
}
}
mHelpItem = menu.findItem(R.id.help_info_menu_item);
mSendFeedbackItem = menu.findItem(R.id.feedback_menu_item);
mFolderSettingsItem = menu.findItem(R.id.folder_options);
mEmptyTrashItem = menu.findItem(R.id.empty_trash);
mEmptySpamItem = menu.findItem(R.id.empty_spam);
return true;
}
public int getOptionsMenuId() {
switch (getMode()) {
case ViewMode.UNKNOWN:
return R.menu.conversation_list_menu;
case ViewMode.CONVERSATION:
return R.menu.conversation_actions;
case ViewMode.CONVERSATION_LIST:
return R.menu.conversation_list_menu;
case ViewMode.SEARCH_RESULTS_LIST:
return R.menu.conversation_list_search_results_actions;
case ViewMode.SEARCH_RESULTS_CONVERSATION:
return R.menu.conversation_actions;
case ViewMode.WAITING_FOR_ACCOUNT_INITIALIZATION:
return R.menu.wait_mode_actions;
}
LogUtils.wtf(LOG_TAG, "Menu requested for unknown view mode");
return R.menu.conversation_list_menu;
}
public void initialize(ControllableActivity activity, ActivityController callback,
ActionBar actionBar) {
mActionBar = actionBar;
mController = callback;
mActivity = activity;
initializeTitleViews();
mFolderObserver = new FolderObserver() {
@Override
public void onChanged(Folder newFolder) {
onFolderUpdated(newFolder);
}
};
// Return values are purposely discarded. Initialization happens quite early, and we don't
// have a valid folder, or a valid list of accounts.
mFolderObserver.initialize(mController);
updateAccount(mAccountObserver.initialize(activity.getAccountController()));
}
private void updateAccount(Account account) {
final boolean accountChanged = mAccount == null || !mAccount.uri.equals(account.uri);
mAccount = account;
if (mAccount != null && accountChanged) {
final ContentResolver resolver = mActivity.getActivityContext().getContentResolver();
final Bundle bundle = new Bundle(1);
bundle.putParcelable(UIProvider.SetCurrentAccountColumns.ACCOUNT, account);
final UpdateProvider updater = new UpdateProvider(mAccount.uri, resolver);
updater.execute(bundle);
setFolderAndAccount(false /* folderChanged */);
}
}
/**
* Called by the owner of the ActionBar to change the current folder.
*/
public void setFolder(Folder folder) {
mFolder = folder;
setFolderAndAccount(true);
}
public void onDestroy() {
if (mFolderObserver != null) {
mFolderObserver.unregisterAndDestroy();
mFolderObserver = null;
}
mAccountObserver.unregisterAndDestroy();
mHandler.removeMessages(SubtitleHandler.EMAIL);
}
@Override
public void onViewModeChanged(int newMode) {
mActivity.invalidateOptionsMenu();
mHandler.removeMessages(SubtitleHandler.EMAIL);
// Check if we are either on a phone, or in Conversation mode on tablet. For these, the
// recent folders is enabled.
switch (getMode()) {
case ViewMode.UNKNOWN:
break;
case ViewMode.CONVERSATION_LIST:
showNavList();
break;
case ViewMode.SEARCH_RESULTS_CONVERSATION:
mActionBar.setDisplayHomeAsUpEnabled(true);
setEmptyMode();
break;
case ViewMode.CONVERSATION:
case ViewMode.AD:
closeSearchField();
mActionBar.setDisplayHomeAsUpEnabled(true);
setEmptyMode();
break;
case ViewMode.WAITING_FOR_ACCOUNT_INITIALIZATION:
// We want the user to be able to switch accounts while waiting for an account
// to sync.
showNavList();
break;
}
}
/**
* Close the search query entry field to avoid keyboard events, and to restore the actionbar
* to non-search mode.
*/
private void closeSearchField() {
if (mSearch == null) {
return;
}
mSearch.collapseActionView();
}
protected int getMode() {
if (mViewModeController != null) {
return mViewModeController.getMode();
} else {
return ViewMode.UNKNOWN;
}
}
public boolean onPrepareOptionsMenu(Menu menu) {
// We start out with every option enabled. Based on the current view, we disable actions
// that are possible.
LogUtils.d(LOG_TAG, "ActionBarView.onPrepareOptionsMenu().");
if (mHelpItem != null) {
mHelpItem.setVisible(mAccount != null
&& mAccount.supportsCapability(AccountCapabilities.HELP_CONTENT));
}
if (mSendFeedbackItem != null) {
mSendFeedbackItem.setVisible(mAccount != null
&& mAccount.supportsCapability(AccountCapabilities.SEND_FEEDBACK));
}
if (mController.shouldHideMenuItems()) {
// Shortcut: hide all remaining menu items if the drawer is shown
final int size = menu.size();
for (int i = 0; i < size; i++) {
final MenuItem item = menu.getItem(i);
final int id = item.getItemId();
if (id != R.id.settings
&& id != R.id.feedback_menu_item
&& id != R.id.help_info_menu_item) {
item.setVisible(false);
}
}
return false;
}
if (mFolderSettingsItem != null) {
mFolderSettingsItem.setVisible(mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.SUPPORTS_SETTINGS));
}
if (mEmptyTrashItem != null) {
mEmptyTrashItem.setVisible(mAccount != null && mFolder != null
&& mAccount.supportsCapability(AccountCapabilities.EMPTY_TRASH)
&& mFolder.isTrash() && mFolder.totalCount > 0);
}
if (mEmptySpamItem != null) {
mEmptySpamItem.setVisible(mAccount != null && mFolder != null
&& mAccount.supportsCapability(AccountCapabilities.EMPTY_SPAM)
&& mFolder.isType(FolderType.SPAM) && mFolder.totalCount > 0);
}
switch (getMode()) {
case ViewMode.CONVERSATION:
case ViewMode.SEARCH_RESULTS_CONVERSATION:
// We update the ActionBar options when we are entering conversation view because
// waiting for the AbstractConversationViewFragment to do it causes duplicate icons
// to show up during the time between the conversation is selected and the fragment
// is added.
setConversationModeOptions(menu);
// We want to use the user's preferred menu items here
final Resources resources = getResources();
final int maxItems = resources.getInteger(R.integer.actionbar_max_items);
final int hiddenItems = resources.getInteger(
R.integer.actionbar_hidden_non_cab_items_no_physical_button);
final int totalItems = maxItems
- (ViewConfiguration.get(getContext()).hasPermanentMenuKey()
? 0 : hiddenItems);
reorderMenu(getContext(), mAccount, menu, totalItems);
break;
case ViewMode.CONVERSATION_LIST:
// Show search if the account supports it
Utils.setMenuItemVisibility(menu, R.id.search, mAccount.supportsSearch());
break;
case ViewMode.SEARCH_RESULTS_LIST:
// Hide compose and search
Utils.setMenuItemVisibility(menu, R.id.compose, false);
Utils.setMenuItemVisibility(menu, R.id.search, false);
break;
}
return false;
}
/**
* Reorders the specified {@link Menu}, taking into account the user's Archive/Delete
* preference.
*/
public static void reorderMenu(final Context context, final Account account, final Menu menu,
final int maxItems) {
final String removalAction = MailPrefs.get(context).getRemovalAction(
account.supportsCapability(AccountCapabilities.ARCHIVE));
final boolean showArchive = MailPrefs.RemovalActions.ARCHIVE.equals(removalAction) ||
MailPrefs.RemovalActions.ARCHIVE_AND_DELETE.equals(removalAction);
final boolean showDelete = MailPrefs.RemovalActions.DELETE.equals(removalAction) ||
MailPrefs.RemovalActions.ARCHIVE_AND_DELETE.equals(removalAction);
// Do a first pass to extract necessary information on what is safe to display
boolean archiveVisibleEnabled = false;
boolean deleteVisibleEnabled = false;
for (int i = 0; i < menu.size(); i++) {
final MenuItem menuItem = menu.getItem(i);
final int itemId = menuItem.getItemId();
final boolean visible = menuItem.isVisible();
final boolean enabled = menuItem.isEnabled();
if (itemId == R.id.archive || itemId == R.id.remove_folder) {
archiveVisibleEnabled |= (visible & enabled);
} else if (itemId == R.id.delete || itemId == R.id.discard_drafts) {
deleteVisibleEnabled |= (visible & enabled);
}
}
int actionItems = 0;
for (int i = 0; i < menu.size(); i++) {
final MenuItem menuItem = menu.getItem(i);
final int itemId = menuItem.getItemId();
// We only want to promote it if it's visible and has an icon
if (menuItem.isVisible() && menuItem.getIcon() != null) {
if (itemId == R.id.archive || itemId == R.id.remove_folder) {
/*
* If this is disabled, and we want to show both archive and delete, we will
* hide archive (rather than showing it disabled), and take up one of our
* spaces. If we only want to show archive, we'll hide it, but not take up
* a space.
*/
if (!menuItem.isEnabled() && showArchive) {
menuItem.setVisible(false);
if (showDelete) {
actionItems++;
}
} else {
/*
* We show this if the following are all true:
* 1. The user wants to display archive, or delete is not visible
* 2. We have room for it
*/
if ((showArchive || !deleteVisibleEnabled) && actionItems < maxItems) {
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
actionItems++;
}
}
} else if (itemId == R.id.delete || itemId == R.id.discard_drafts) {
/*
* We show this if the following are all true:
* 1. The user wants to display delete, or archive is not visible
* 2. We have room for it
*/
if ((showDelete || !archiveVisibleEnabled) && actionItems < maxItems) {
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
actionItems++;
}
} else if (itemId == R.id.change_folders) {
final boolean showChangeFolder = account
.supportsCapability(AccountCapabilities.MULTIPLE_FOLDERS_PER_CONV);
menuItem.setVisible(showChangeFolder);
if (showChangeFolder && actionItems < maxItems) {
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
actionItems++;
}
} else if (itemId == R.id.search) {
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS
| MenuItem.SHOW_AS_ACTION_COLLAPSE_ACTION_VIEW);
actionItems++;
} else {
if (actionItems < maxItems) {
menuItem.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS);
actionItems++;
}
}
}
}
}
/**
* Put the ActionBar in List navigation mode.
*/
private void showNavList() {
setTitleModeFlags(getActionBarTitleModeFlag());
setFolderAndAccount(false);
}
private void setSubtitle(CharSequence subtitle) {
if (!TextUtils.equals(subtitle, mActionBar.getSubtitle())) {
mActionBar.setSubtitle(subtitle);
}
if (mLegacySubTitle != null) {
mLegacySubTitle.setText(subtitle);
}
}
private void setTitle(String title) {
title = mBidiFormatter.unicodeWrap(title);
if (!TextUtils.equals(title, mActionBar.getTitle())) {
mActionBar.setTitle(title);
}
if (mLegacyTitle != null) {
mLegacyTitle.setText(title);
}
}
private int getActionBarTitleModeFlag() {
return mUseLegacyTitle ? ActionBar.DISPLAY_SHOW_CUSTOM : ActionBar.DISPLAY_SHOW_TITLE;
}
/**
* Set the actionbar mode to empty: no title, no subtitle, no custom view.
*/
protected void setEmptyMode() {
// Disable title/subtitle and the custom view by setting the bitmask to all off.
setTitleModeFlags(0);
}
/**
* Removes the back button from being shown
*/
public void removeBackButton() {
if (mActionBar == null) {
return;
}
// Remove the back button but continue showing an icon.
final int mask = ActionBar.DISPLAY_HOME_AS_UP | ActionBar.DISPLAY_SHOW_HOME;
mActionBar.setDisplayOptions(ActionBar.DISPLAY_SHOW_HOME, mask);
mActivity.getActionBar().setHomeButtonEnabled(false);
}
public void setBackButton() {
if (mActionBar == null) {
return;
}
// Show home as up, and show an icon.
final int mask = ActionBar.DISPLAY_HOME_AS_UP | ActionBar.DISPLAY_SHOW_HOME;
mActionBar.setDisplayOptions(mask, mask);
mActivity.getActionBar().setHomeButtonEnabled(true);
}
@Override
public boolean onQueryTextSubmit(String query) {
if (mSearch != null) {
mSearch.collapseActionView();
mSearchWidget.setQuery("", false);
}
mController.executeSearch(query.trim());
return true;
}
@Override
public boolean onQueryTextChange(String newText) {
return false;
}
// Next two methods are called when search suggestions are clicked.
@Override
public boolean onSuggestionSelect(int position) {
return onSuggestionClick(position);
}
@Override
public boolean onSuggestionClick(int position) {
final Cursor c = mSearchWidget.getSuggestionsAdapter().getCursor();
final boolean haveValidQuery = (c != null) && c.moveToPosition(position);
if (!haveValidQuery) {
LogUtils.d(LOG_TAG, "onSuggestionClick: Couldn't get a search query");
// We haven't handled this query, but the default behavior will
// leave EXTRA_ACCOUNT un-populated, leading to a crash. So claim
// that we have handled the event.
return true;
}
collapseSearch();
// what is in the text field
String queryText = mSearchWidget.getQuery().toString();
// What the suggested query is
String query = c.getString(c.getColumnIndex(SearchManager.SUGGEST_COLUMN_QUERY));
// If the text the user typed in is a prefix of what is in the search
// widget suggestion query, just take the search widget suggestion
// query. Otherwise, it is a suffix and we want to remove matching
// prefix portions.
if (!TextUtils.isEmpty(queryText) && query.indexOf(queryText) != 0) {
final int queryTokenIndex = queryText
.lastIndexOf(SearchRecentSuggestionsProvider.QUERY_TOKEN_SEPARATOR);
if (queryTokenIndex > -1) {
queryText = queryText.substring(0, queryTokenIndex);
}
// Since we auto-complete on each token in a query, if the query the
// user typed up until the last token is a substring of the
// suggestion they click, make sure we don't double include the
// query text. For example:
// user types john, that matches john palo alto
// User types john p, that matches john john palo alto
// Remove the first john
// Only do this if we have multiple query tokens.
if (queryTokenIndex > -1 && !TextUtils.isEmpty(query) && query.contains(queryText)
&& queryText.length() < query.length()) {
int start = query.indexOf(queryText);
query = query.substring(0, start) + query.substring(start + queryText.length());
}
}
mController.executeSearch(query.trim());
return true;
}
/**
* Uses the current state to update the current folder {@link #mFolder} and the current
* account {@link #mAccount} shown in the actionbar. Also updates the actionbar subtitle to
* momentarily display the unread count if it has changed.
* @param folderChanged true if folder changed in terms of URI
*/
private void setFolderAndAccount(final boolean folderChanged) {
// Very little can be done if the actionbar or activity is null.
if (mActionBar == null || mActivity == null) {
return;
}
if (ViewMode.isWaitingForSync(getMode())) {
// Account is not synced: clear title and update the subtitle.
setTitle("");
removeUnreadCount(true);
return;
}
// Check if we should be changing the actionbar at all, and back off if not.
final boolean isShowingFolder = mIsOnTablet || ViewMode.isListMode(getMode());
if (!isShowingFolder) {
// It isn't necessary to set the title in this case, as the title view will
// be hidden
return;
}
if (mFolder == null) {
// Clear the action bar title. We don't want the app name to be shown while
// waiting for the folder query to finish
setTitle("");
return;
}
setTitle(mFolder.name);
final int folderUnreadCount = mFolder.isUnreadCountHidden() ? 0 : mFolder.unreadCount;
// The user shouldn't see "999+ unread messages", and then a short while later: "999+
// unread messages". So we set our unread count just past the limit. This way we can
// change the subtitle the first time around but not for subsequent changes as far as the
// unread count remains over the limit.
final int toDisplay = (folderUnreadCount > UNREAD_LIMIT)
? (UNREAD_LIMIT + 1) : folderUnreadCount;
if ((mUnreadCount != toDisplay || folderChanged) && toDisplay != 0) {
setSubtitle(Utils.getUnreadMessageString(mActivity.getApplicationContext(), toDisplay));
}
// Schedule a removal of unread count for the future, if there isn't one already. If the
// unread count dropped to zero, remove it and show the account name right away.
removeUnreadCount(toDisplay == 0);
// Remember the new value for the next run
mUnreadCount = toDisplay;
}
/**
* Remove the unread count and show the account name, if required.
* @param now true if you want the change to happen immediately. False if you want to enforce
* it happens later.
*/
private void removeUnreadCount(boolean now) {
if (now) {
// Remove all previous messages which might change the subtitle
mHandler.removeMessages(SubtitleHandler.EMAIL);
// Update the subtitle: clear it or show account name.
mHandler.sendEmptyMessage(SubtitleHandler.EMAIL);
} else {
if (!mHandler.hasMessages(SubtitleHandler.EMAIL)) {
// In a short while, show the account name in its place.
mHandler.sendEmptyMessageDelayed(SubtitleHandler.EMAIL, ACCOUNT_DELAY_MS);
}
}
}
/**
* Notify that the folder has changed.
*/
public void onFolderUpdated(Folder folder) {
if (folder == null) {
return;
}
/** True if we are changing folders. */
final boolean changingFolders = (mFolder == null || !mFolder.equals(folder));
mFolder = folder;
setFolderAndAccount(changingFolders);
final ConversationListContext listContext = mController == null ? null :
mController.getCurrentListContext();
if (changingFolders && !ConversationListContext.isSearchResult(listContext)) {
closeSearchField();
}
}
@Override
public boolean onMenuItemActionExpand(MenuItem item) {
// Do nothing. Required as part of the interface, we ar only interested in
// onMenuItemActionCollapse(MenuItem).
// Have to return true here. Unlike other callbacks, the return value here is whether
// we want to suppress the action (rather than consume the action). We don't want to
// suppress the action.
return true;
}
@Override
public boolean onMenuItemActionCollapse(MenuItem item) {
// Work around b/6664203 by manually forcing this view to be VISIBLE
// upon ActionView collapse. DISPLAY_SHOW_CUSTOM will still control its final
// visibility.
setVisibility(VISIBLE);
// Have to return true here. Unlike other callbacks, the return value
// here is whether we want to suppress the action (rather than consume the action). We
// don't want to suppress the action.
return true;
}
/**
* Sets the actionbar mode: Pass it an integer which contains each of these values, perhaps
* OR'd together: {@link ActionBar#DISPLAY_SHOW_CUSTOM} and
* {@link ActionBar#DISPLAY_SHOW_TITLE}. To disable all, pass a zero.
* @param enabledFlags
*/
private void setTitleModeFlags(int enabledFlags) {
final int mask = ActionBar.DISPLAY_SHOW_TITLE | ActionBar.DISPLAY_SHOW_CUSTOM;
mActionBar.setDisplayOptions(enabledFlags, mask);
}
public void setCurrentConversation(Conversation conversation) {
mCurrentConversation = conversation;
}
//We need to do this here instead of in the fragment
public void setConversationModeOptions(Menu menu) {
if (mCurrentConversation == null) {
return;
}
final boolean showMarkImportant = !mCurrentConversation.isImportant();
Utils.setMenuItemVisibility(menu, R.id.mark_important, showMarkImportant
&& mAccount.supportsCapability(UIProvider.AccountCapabilities.MARK_IMPORTANT));
Utils.setMenuItemVisibility(menu, R.id.mark_not_important, !showMarkImportant
&& mAccount.supportsCapability(UIProvider.AccountCapabilities.MARK_IMPORTANT));
final boolean showDelete = mFolder != null &&
mFolder.supportsCapability(UIProvider.FolderCapabilities.DELETE);
Utils.setMenuItemVisibility(menu, R.id.delete, showDelete);
// We only want to show the discard drafts menu item if we are not showing the delete menu
// item, and the current folder is a draft folder and the account supports discarding
// drafts for a conversation
final boolean showDiscardDrafts = !showDelete && mFolder != null && mFolder.isDraft() &&
mAccount.supportsCapability(AccountCapabilities.DISCARD_CONVERSATION_DRAFTS);
Utils.setMenuItemVisibility(menu, R.id.discard_drafts, showDiscardDrafts);
final boolean archiveVisible = mAccount.supportsCapability(AccountCapabilities.ARCHIVE)
&& mFolder != null && mFolder.supportsCapability(FolderCapabilities.ARCHIVE)
&& !mFolder.isTrash();
Utils.setMenuItemVisibility(menu, R.id.archive, archiveVisible);
Utils.setMenuItemVisibility(menu, R.id.remove_folder, !archiveVisible && mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.CAN_ACCEPT_MOVED_MESSAGES)
&& !mFolder.isProviderFolder()
&& mAccount.supportsCapability(AccountCapabilities.ARCHIVE));
Utils.setMenuItemVisibility(menu, R.id.move_to, mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.ALLOWS_REMOVE_CONVERSATION));
Utils.setMenuItemVisibility(menu, R.id.move_to_inbox, mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.ALLOWS_MOVE_TO_INBOX));
final MenuItem removeFolder = menu.findItem(R.id.remove_folder);
if (mFolder != null && removeFolder != null) {
removeFolder.setTitle(mActivity.getApplicationContext().getString(
R.string.remove_folder, mFolder.name));
}
Utils.setMenuItemVisibility(menu, R.id.report_spam,
mAccount.supportsCapability(AccountCapabilities.REPORT_SPAM) && mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.REPORT_SPAM)
&& !mCurrentConversation.spam);
Utils.setMenuItemVisibility(menu, R.id.mark_not_spam,
mAccount.supportsCapability(AccountCapabilities.REPORT_SPAM) && mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.MARK_NOT_SPAM)
&& mCurrentConversation.spam);
Utils.setMenuItemVisibility(menu, R.id.report_phishing,
mAccount.supportsCapability(AccountCapabilities.REPORT_PHISHING) && mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.REPORT_PHISHING)
&& !mCurrentConversation.phishing);
Utils.setMenuItemVisibility(menu, R.id.mute,
mAccount.supportsCapability(AccountCapabilities.MUTE) && mFolder != null
&& mFolder.supportsCapability(FolderCapabilities.DESTRUCTIVE_MUTE)
&& !mCurrentConversation.muted);
}
private static boolean actionBarSupportsNewMethods(ActionBar bar) {
// TODO(pwestbro) switch this to
// (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) when we switch to the
// latest SDK
if (Build.VERSION.SDK_INT > 17) {
return true;
}
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.JELLY_BEAN) {
return false;
}
boolean supportsNewApi = false;
try {
if (bar != null) {
supportsNewApi = (ActionBar.class.getField("DISPLAY_TITLE_MULTIPLE_LINES") != null);
}
} catch (NoSuchFieldException e) {
// stay false
}
return supportsNewApi;
}
@Override
public void onClick (View v) {
if (v.getId() == R.id.legacy_title_container) {
mController.onUpPressed();
}
}
public ViewMode getViewModeController() {
return mViewModeController;
}
public void setViewModeController(ViewMode viewModeController) {
mViewModeController = viewModeController;
mViewModeController.addListener(this);
}
}