/*
* Copyright (C) 2008 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.email.activity;
import android.app.Activity;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.text.TextUtils;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import com.android.email.Email;
import com.android.email.Preferences;
import com.android.email.R;
import com.android.email.activity.setup.AccountSettings;
import com.android.email.activity.setup.AccountSetupBasics;
import com.android.email.service.EmailServiceUtils;
import com.android.email.service.MailService;
import com.android.emailcommon.Logging;
import com.android.emailcommon.provider.Account;
import com.android.emailcommon.provider.EmailContent;
import com.android.emailcommon.provider.EmailContent.Message;
import com.android.emailcommon.provider.Mailbox;
import com.android.emailcommon.utility.EmailAsyncTask;
import com.android.emailcommon.utility.IntentUtilities;
import com.android.emailcommon.utility.Utility;
import com.google.common.annotations.VisibleForTesting;
/**
* The Welcome activity initializes the application and starts {@link EmailActivity}, or launch
* {@link AccountSetupBasics} if no accounts are configured.
*
* TOOD Show "your messages are on the way" message like gmail does during the inbox lookup.
*/
public class Welcome extends Activity {
/*
* Commands for testing...
* Open 1 pane
adb shell am start -a android.intent.action.MAIN \
-d '"content://ui.email.android.com/view/mailbox"' \
-e DEBUG_PANE_MODE 1
* Open 2 pane
adb shell am start -a android.intent.action.MAIN \
-d '"content://ui.email.android.com/view/mailbox"' \
-e DEBUG_PANE_MODE 2
* Open an account (ID=1) in 2 pane
adb shell am start -a android.intent.action.MAIN \
-d '"content://ui.email.android.com/view/mailbox?ACCOUNT_ID=1"' \
-e DEBUG_PANE_MODE 2
* Open a message (account id=1, mailbox id=2, message id=3)
adb shell am start -a android.intent.action.MAIN \
-d '"content://ui.email.android.com/view/mailbox?ACCOUNT_ID=1&MAILBOX_ID=2&MESSAGE_ID=3"' \
-e DEBUG_PANE_MODE 2
* Open the combined starred on the combined view
adb shell am start -a android.intent.action.MAIN \
-d '"content://ui.email.android.com/view/mailbox?ACCOUNT_ID=1152921504606846976&MAILBOX_ID=-4"' \
-e DEBUG_PANE_MODE 2
*/
/**
* Extra for debugging. Set 1 to force one-pane. Set 2 to force two-pane.
*/
private static final String EXTRA_DEBUG_PANE_MODE = "DEBUG_PANE_MODE";
private static final String VIEW_MAILBOX_INTENT_URL_PATH = "/view/mailbox";
private final EmailAsyncTask.Tracker mTaskTracker = new EmailAsyncTask.Tracker();
private View mWaitingForSyncView;
private long mAccountId;
private long mMailboxId;
private long mMessageId;
private String mAccountUuid;
private MailboxFinder mInboxFinder;
/**
* Launch this activity. Note: It's assumed that this activity is only called as a means to
* 'reset' the UI state; Because of this, it is always launched with FLAG_ACTIVITY_CLEAR_TOP,
* which will drop any other activities on the stack (e.g. AccountFolderList or MessageList).
*/
public static void actionStart(Activity fromActivity) {
Intent i = IntentUtilities.createRestartAppIntent(fromActivity, Welcome.class);
fromActivity.startActivity(i);
}
/**
* Create an Intent to open email activity. If accountId
is not -1, the
* specified account will be automatically be opened when the activity starts.
*/
public static Intent createOpenAccountInboxIntent(Context context, long accountId) {
final Uri.Builder b = IntentUtilities.createActivityIntentUrlBuilder(
VIEW_MAILBOX_INTENT_URL_PATH);
IntentUtilities.setAccountId(b, accountId);
return IntentUtilities.createRestartAppIntent(b.build());
}
/**
* Create an Intent to open a message.
*/
public static Intent createOpenMessageIntent(Context context, long accountId,
long mailboxId, long messageId) {
final Uri.Builder b = IntentUtilities.createActivityIntentUrlBuilder(
VIEW_MAILBOX_INTENT_URL_PATH);
IntentUtilities.setAccountId(b, accountId);
IntentUtilities.setMailboxId(b, mailboxId);
IntentUtilities.setMessageId(b, messageId);
return IntentUtilities.createRestartAppIntent(b.build());
}
/**
* Open account's inbox.
*/
public static void actionOpenAccountInbox(Activity fromActivity, long accountId) {
fromActivity.startActivity(createOpenAccountInboxIntent(fromActivity, accountId));
}
/**
* Create an {@link Intent} for account shortcuts. The returned intent stores the account's
* UUID rather than the account ID, which will be changed after account restore.
*/
public static Intent createAccountShortcutIntent(Context context, String uuid, long mailboxId) {
final Uri.Builder b = IntentUtilities.createActivityIntentUrlBuilder(
VIEW_MAILBOX_INTENT_URL_PATH);
IntentUtilities.setAccountUuid(b, uuid);
IntentUtilities.setMailboxId(b, mailboxId);
return IntentUtilities.createRestartAppIntent(b.build());
}
/**
* If the {@link #EXTRA_DEBUG_PANE_MODE} extra is "1" or "2", return 1 or 2 respectively.
* Otherwise return 0.
*
* @see UiUtilities#setDebugPaneMode(int)
* @see UiUtilities#useTwoPane(Context)
*/
private static int getDebugPaneMode(Intent i) {
Bundle extras = i.getExtras();
if (extras != null) {
String s = extras.getString(EXTRA_DEBUG_PANE_MODE);
if ("1".equals(s)) {
return 1;
} else if ("2".equals(s)) {
return 2;
}
}
return 0;
}
@Override
public void onCreate(Bundle icicle) {
super.onCreate(icicle);
ActivityHelper.debugSetWindowFlags(this);
// Because the app could be reloaded (for debugging, etc.), we need to make sure that
// ExchangeService gets a chance to start. There is no harm to starting it if it has
// already been started
// When the service starts, it reconciles EAS accounts.
// TODO More completely separate ExchangeService from Email app
EmailServiceUtils.startExchangeService(this);
// Extract parameters from the intent.
final Intent intent = getIntent();
mAccountId = IntentUtilities.getAccountIdFromIntent(intent);
mMailboxId = IntentUtilities.getMailboxIdFromIntent(intent);
mMessageId = IntentUtilities.getMessageIdFromIntent(intent);
mAccountUuid = IntentUtilities.getAccountUuidFromIntent(intent);
UiUtilities.setDebugPaneMode(getDebugPaneMode(intent));
// Reconcile POP/IMAP accounts. EAS accounts are taken care of by ExchangeService.
EmailAsyncTask.runAsyncParallel(new Runnable() {
@Override
public void run() {
// Reconciling can be heavy - so do it in the background.
if (MailService.hasMismatchInPopImapAccounts(Welcome.this)) {
MailService.reconcilePopImapAccountsSync(Welcome.this);
}
Welcome.this.runOnUiThread(new Runnable() {
@Override
public void run() {
resolveAccount();
}});
}
});
// Reset the "accounts changed" notification, now that we're here
Email.setNotifyUiAccountsChanged(false);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Only create the menu if we had to stop and show a loading spinner - otherwise
// this is a transient activity with no UI.
if (mInboxFinder == null) {
return super.onCreateOptionsMenu(menu);
}
getMenuInflater().inflate(R.menu.welcome, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
if (item.getItemId() == R.id.account_settings) {
AccountSettings.actionSettings(this, mAccountId);
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
protected void onStop() {
// Cancel all running tasks.
// (If it's stopping for configuration changes, we just re-do everything on the new
// instance)
stopInboxLookup();
mTaskTracker.cancellAllInterrupt();
super.onStop();
if (!isChangingConfigurations()) {
// This means the user opened some other app.
// Just close self and not launch EmailActivity.
if (Email.DEBUG && Logging.DEBUG_LIFECYCLE) {
Log.d(Logging.LOG_TAG, "Welcome: Closing self...");
}
finish();
}
}
/**
* {@inheritDoc}
*
* When launching an activity from {@link Welcome}, we always want to set
* {@link Intent#FLAG_ACTIVITY_FORWARD_RESULT}.
*/
@Override
public void startActivity(Intent intent) {
intent.setFlags(Intent.FLAG_ACTIVITY_FORWARD_RESULT);
super.startActivity(intent);
}
/**
* Stop inbox lookup. This MSUT be called on the UI thread.
*/
private void stopInboxLookup() {
if (mInboxFinder != null) {
mInboxFinder.cancel();
mInboxFinder = null;
}
}
/**
* Start inbox lookup. This MSUT be called on the UI thread.
*/
private void startInboxLookup() {
Log.i(Logging.LOG_TAG, "Inbox not found. Starting mailbox finder...");
stopInboxLookup(); // Stop if already running -- it shouldn't be but just in case.
mInboxFinder = new MailboxFinder(this, mAccountId, Mailbox.TYPE_INBOX,
mMailboxFinderCallback);
mInboxFinder.startLookup();
// Show "your email will appear shortly" message.
mWaitingForSyncView = LayoutInflater.from(this).inflate(
R.layout.waiting_for_sync_message, null);
addContentView(mWaitingForSyncView, new LayoutParams(
LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
invalidateOptionsMenu();
}
/**
* Determine which account to open with the given account ID and UUID.
*
* @return ID of the account to use.
*/
@VisibleForTesting
static long resolveAccountId(Context context, long inputAccountId, String inputUuid) {
final long accountId;
if (!TextUtils.isEmpty(inputUuid)) {
// If a UUID is specified, try to use it.
// If the UUID is invalid, accountId will be NO_ACCOUNT.
accountId = Account.getAccountIdFromUuid(context, inputUuid);
} else if (inputAccountId != Account.NO_ACCOUNT) {
// If a valid account ID is specified, just use it.
if (inputAccountId == Account.ACCOUNT_ID_COMBINED_VIEW
|| Account.isValidId(context, inputAccountId)) {
accountId = inputAccountId;
} else {
accountId = Account.NO_ACCOUNT;
}
} else {
// Neither an accountID or a UUID is specified.
// Use the last account used, falling back to the default.
long lastUsedId = Preferences.getPreferences(context).getLastUsedAccountId();
if (lastUsedId != Account.NO_ACCOUNT) {
if (!Account.isValidId(context, lastUsedId)) {
// The last account that was used has since been deleted.
lastUsedId = Account.NO_ACCOUNT;
Preferences.getPreferences(context).setLastUsedAccountId(Account.NO_ACCOUNT);
}
}
accountId = (lastUsedId == Account.NO_ACCOUNT)
? Account.getDefaultAccountId(context)
: lastUsedId;
}
if (accountId != Account.NO_ACCOUNT) {
// Okay, the given account is valid.
return accountId;
} else {
// No, it's invalid. Show the warning toast and use the default.
Utility.showToast(context, R.string.toast_account_not_found);
return Account.getDefaultAccountId(context);
}
}
/**
* Determine which account to use according to the number of accounts already set up,
* {@link #mAccountId} and {@link #mAccountUuid}.
*
*
* 1. If there's no account configured, start account setup. * 2. Otherwise detemine which account to open with {@link #resolveAccountId} and * 2a. If the account doesn't have inbox yet, start inbox finder. * 2b. Otherwise open the main activity. **/ private void resolveAccount() { final int numAccount = EmailContent.count(this, Account.CONTENT_URI); if (numAccount == 0) { AccountSetupBasics.actionNewAccount(this); finish(); return; } else { mAccountId = resolveAccountId(this, mAccountId, mAccountUuid); if (Account.isNormalAccount(mAccountId) && Mailbox.findMailboxOfType(this, mAccountId, Mailbox.TYPE_INBOX) == Mailbox.NO_MAILBOX) { startInboxLookup(); return; } } startEmailActivity(); } /** * Start {@link EmailActivity} using {@link #mAccountId}, {@link #mMailboxId} and * {@link #mMessageId}. */ private void startEmailActivity() { final Intent i; if (mMessageId != Message.NO_MESSAGE) { i = EmailActivity.createOpenMessageIntent(this, mAccountId, mMailboxId, mMessageId); } else if (mMailboxId != Mailbox.NO_MAILBOX) { i = EmailActivity.createOpenMailboxIntent(this, mAccountId, mMailboxId); } else { i = EmailActivity.createOpenAccountIntent(this, mAccountId); } startActivity(i); finish(); } private final MailboxFinder.Callback mMailboxFinderCallback = new MailboxFinder.Callback() { // This MUST be called from callback methods. private void cleanUp() { mInboxFinder = null; } @Override public void onAccountNotFound() { cleanUp(); // Account removed? Clear the IDs and restart the task. Which will result in either // a) show account setup if there's really no accounts or b) open the default account. mAccountId = Account.NO_ACCOUNT; mMailboxId = Mailbox.NO_MAILBOX; mMessageId = Message.NO_MESSAGE; mAccountUuid = null; // Restart the account resolution. resolveAccount(); } @Override public void onMailboxNotFound(long accountId) { // Just do the same thing as "account not found". onAccountNotFound(); } @Override public void onAccountSecurityHold(long accountId) { cleanUp(); ActivityHelper.showSecurityHoldDialog(Welcome.this, accountId); finish(); } @Override public void onMailboxFound(long accountId, long mailboxId) { cleanUp(); // Okay the account has Inbox now. Start the main activity. startEmailActivity(); } }; }