| /* |
| * Copyright (C) 2013 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.animation.Animator; |
| import android.animation.ObjectAnimator; |
| import android.animation.Animator.AnimatorListener; |
| import android.app.Activity; |
| import android.app.LoaderManager; |
| import android.content.ContentResolver; |
| import android.content.Context; |
| import android.content.res.Resources; |
| import android.os.Bundle; |
| import android.text.SpannableString; |
| import android.text.TextUtils; |
| import android.text.style.TextAppearanceSpan; |
| import android.util.AttributeSet; |
| import android.view.View; |
| import android.view.animation.DecelerateInterpolator; |
| import android.widget.FrameLayout; |
| import android.widget.TextView; |
| |
| import com.android.mail.R; |
| import com.android.mail.analytics.Analytics; |
| import com.android.mail.browse.ConversationCursor; |
| import com.android.mail.preferences.AccountPreferences; |
| import com.android.mail.preferences.MailPrefs; |
| import com.android.mail.providers.Account; |
| import com.android.mail.providers.Folder; |
| import com.android.mail.utils.LogTag; |
| import com.android.mail.utils.LogUtils; |
| import com.android.mail.utils.Utils; |
| |
| /** |
| * A tip displayed on top of conversation view to indicate that Gmail sync is |
| * currently disabled on this account. |
| */ |
| public class ConversationSyncDisabledTipView extends FrameLayout |
| implements ConversationSpecialItemView, SwipeableItemView { |
| |
| private static final String LOG_TAG = LogTag.getLogTag(); |
| |
| private static int sScrollSlop = 0; |
| private static int sShrinkAnimationDuration; |
| |
| private Account mAccount = null; |
| private Folder mFolder = null; |
| private final MailPrefs mMailPrefs; |
| private AccountPreferences mAccountPreferences; |
| private AnimatedAdapter mAdapter; |
| private Activity mActivity; |
| |
| private View mSwipeableContent; |
| private TextView mText1; |
| private TextView mText2; |
| private View mTextArea; |
| private SpannableString mEnableSyncInAccountSettingsText; |
| private final OnClickListener mAutoSyncOffTextClickedListener; |
| private final OnClickListener mAccountSyncOffTextClickedListener; |
| |
| private int mAnimatedHeight = -1; |
| |
| private int mReasonSyncOff = ReasonSyncOff.NONE; |
| |
| private View mTeaserRightEdge; |
| /** Whether we are on a tablet device or not */ |
| private final boolean mTabletDevice; |
| /** When in conversation mode, true if the list is hidden */ |
| private final boolean mListCollapsible; |
| |
| public interface ReasonSyncOff { |
| // Background sync is enabled for current account, do not display this tip |
| public static final int NONE = 0; |
| // Global auto-sync (affects all apps and all accounts) is turned off |
| public static final int AUTO_SYNC_OFF = 1; |
| // Global auto-sync is on, but Gmail app level sync is disabled for this particular account |
| public static final int ACCOUNT_SYNC_OFF = 2; |
| } |
| |
| public ConversationSyncDisabledTipView(final Context context) { |
| this(context, null); |
| } |
| |
| public ConversationSyncDisabledTipView(final Context context, final AttributeSet attrs) { |
| this(context, attrs, -1); |
| } |
| |
| public ConversationSyncDisabledTipView( |
| final Context context, final AttributeSet attrs, final int defStyle) { |
| super(context, attrs, defStyle); |
| |
| final Resources resources = context.getResources(); |
| |
| if (sScrollSlop == 0) { |
| sScrollSlop = resources.getInteger(R.integer.swipeScrollSlop); |
| sShrinkAnimationDuration = resources.getInteger( |
| R.integer.shrink_animation_duration); |
| } |
| |
| mMailPrefs = MailPrefs.get(context); |
| |
| mAutoSyncOffTextClickedListener = new OnClickListener() { |
| @Override |
| public void onClick(View v) { |
| final TurnAutoSyncOnDialog dialog = TurnAutoSyncOnDialog.newInstance( |
| mAccount.getAccountManagerAccount(), mAccount.syncAuthority); |
| dialog.show(mActivity.getFragmentManager(), TurnAutoSyncOnDialog.DIALOG_TAG); |
| } |
| }; |
| |
| mAccountSyncOffTextClickedListener = new OnClickListener() { |
| @Override |
| public void onClick(View v) { |
| Utils.showAccountSettings(getContext(), mAccount); |
| } |
| }; |
| |
| // Create the "Turn on in Account settings." text where "Account settings" appear as |
| // a blue link. |
| final String subString = resources.getString(R.string.account_settings_param); |
| final String entireString = resources.getString( |
| R.string.enable_sync_in_account_settings, subString); |
| mEnableSyncInAccountSettingsText = new SpannableString(entireString); |
| final int index = entireString.indexOf(subString); |
| mEnableSyncInAccountSettingsText.setSpan( |
| new TextAppearanceSpan(context, R.style.LinksInTipTextAppearance), |
| index, |
| index + subString.length(), |
| 0); |
| |
| mTabletDevice = Utils.useTabletUI(resources); |
| mListCollapsible = resources.getBoolean(R.bool.list_collapsible); |
| } |
| |
| public void bindAccount(Account account, ControllableActivity activity) { |
| mAccount = account; |
| mAccountPreferences = AccountPreferences.get(getContext(), account.getEmailAddress()); |
| mActivity = (Activity) activity; |
| } |
| |
| @Override |
| public void onGetView() { |
| // Do nothing |
| } |
| |
| @Override |
| protected void onFinishInflate() { |
| mSwipeableContent = findViewById(R.id.swipeable_content); |
| |
| mText1 = (TextView) findViewById(R.id.text_line1); |
| mText2 = (TextView) findViewById(R.id.text_line2); |
| mTextArea = findViewById(R.id.text_area); |
| |
| findViewById(R.id.dismiss_button).setOnClickListener(new OnClickListener() { |
| @Override |
| public void onClick(View v) { |
| dismiss(); |
| } |
| }); |
| |
| mTeaserRightEdge = findViewById(R.id.teaser_right_edge); |
| } |
| |
| @Override |
| public void onUpdate(Folder folder, ConversationCursor cursor) { |
| mFolder = folder; |
| } |
| |
| @Override |
| public boolean getShouldDisplayInList() { |
| if (mAccount == null || mAccount.syncAuthority == null) { |
| return false; |
| } |
| |
| // Do not show this message for folders/labels that are not set to sync. |
| if (mFolder == null || mFolder.syncWindow <= 0) { |
| return false; |
| } |
| |
| setReasonSyncOff(calculateReasonSyncOff(mMailPrefs, mAccount, mAccountPreferences)); |
| |
| if (mReasonSyncOff != ReasonSyncOff.NONE) { |
| LogUtils.i(LOG_TAG, "Sync is off with reason %d", mReasonSyncOff); |
| } |
| |
| switch (mReasonSyncOff) { |
| case ReasonSyncOff.AUTO_SYNC_OFF: |
| return (mMailPrefs.getNumOfDismissesForAutoSyncOff() == 0); |
| case ReasonSyncOff.ACCOUNT_SYNC_OFF: |
| return (mAccountPreferences.getNumOfDismissesForAccountSyncOff() == 0); |
| default: |
| return false; |
| } |
| } |
| |
| public static int calculateReasonSyncOff(MailPrefs mailPrefs, |
| Account account, AccountPreferences accountPreferences) { |
| if (!ContentResolver.getMasterSyncAutomatically()) { |
| // Global sync is turned off |
| accountPreferences.resetNumOfDismissesForAccountSyncOff(); |
| // Logging to track down bug where this tip is being showing when it shouldn't be. |
| LogUtils.i(LOG_TAG, "getMasterSyncAutomatically() return false"); |
| return ReasonSyncOff.AUTO_SYNC_OFF; |
| } else { |
| // Global sync is on, clear the number of times users has dismissed this |
| // warning so that next time global sync is off, warning gets displayed again. |
| mailPrefs.resetNumOfDismissesForAutoSyncOff(); |
| |
| // Now check for whether account level sync is on/off. |
| android.accounts.Account acct = account.getAccountManagerAccount(); |
| if (!TextUtils.isEmpty(account.syncAuthority) && |
| !ContentResolver.getSyncAutomatically(acct, account.syncAuthority)) { |
| // Account level sync is off |
| return ReasonSyncOff.ACCOUNT_SYNC_OFF; |
| } else { |
| // Account sync is on, clear the number of times users has dismissed this |
| // warning so that next time sync is off, warning gets displayed again. |
| accountPreferences.resetNumOfDismissesForAccountSyncOff(); |
| return ReasonSyncOff.NONE; |
| } |
| } |
| } |
| |
| private void setReasonSyncOff(int reason) { |
| if (mReasonSyncOff != reason) { |
| mReasonSyncOff = reason; |
| switch (mReasonSyncOff) { |
| case ReasonSyncOff.AUTO_SYNC_OFF: |
| mText1.setText(R.string.auto_sync_off); |
| mText2.setText(R.string.tap_to_enable_sync); |
| mText2.setVisibility(View.VISIBLE); |
| mTextArea.setClickable(true); |
| mTextArea.setOnClickListener(mAutoSyncOffTextClickedListener); |
| break; |
| case ReasonSyncOff.ACCOUNT_SYNC_OFF: |
| mText1.setText(R.string.account_sync_off); |
| mText2.setText(mEnableSyncInAccountSettingsText); |
| mText2.setVisibility(View.VISIBLE); |
| mTextArea.setClickable(true); |
| mTextArea.setOnClickListener(mAccountSyncOffTextClickedListener); |
| break; |
| default: |
| // Doesn't matter what mText is since this view is not displayed |
| } |
| } |
| } |
| |
| @Override |
| public int getPosition() { |
| // We want this teaser to go before the first real conversation |
| return 0; |
| } |
| |
| @Override |
| public void setAdapter(AnimatedAdapter adapter) { |
| mAdapter = adapter; |
| } |
| |
| @Override |
| public void bindFragment(LoaderManager loaderManager, final Bundle savedInstanceState) { |
| } |
| |
| @Override |
| public void cleanup() { |
| } |
| |
| @Override |
| public void onConversationSelected() { |
| // DO NOTHING |
| } |
| |
| @Override |
| public void onCabModeEntered() { |
| } |
| |
| @Override |
| public void onCabModeExited() { |
| // Do nothing |
| } |
| |
| @Override |
| public void onConversationListVisibilityChanged(final boolean visible) { |
| // Do nothing |
| } |
| |
| @Override |
| public void saveInstanceState(final Bundle outState) { |
| // Do nothing |
| } |
| |
| @Override |
| public boolean acceptsUserTaps() { |
| return true; |
| } |
| |
| @Override |
| public void dismiss() { |
| final String reason; |
| switch (mReasonSyncOff) { |
| case ReasonSyncOff.AUTO_SYNC_OFF: |
| mMailPrefs.incNumOfDismissesForAutoSyncOff(); |
| reason = "auto_sync_off"; |
| break; |
| case ReasonSyncOff.ACCOUNT_SYNC_OFF: |
| mAccountPreferences.incNumOfDismissesForAccountSyncOff(); |
| reason = "account_sync_off"; |
| break; |
| default: |
| reason = null; |
| break; |
| } |
| Analytics.getInstance().sendEvent("list_swipe", "sync_disabled_tip", reason, 0); |
| startDestroyAnimation(); |
| } |
| |
| @Override |
| public SwipeableView getSwipeableView() { |
| return SwipeableView.from(mSwipeableContent); |
| } |
| |
| @Override |
| public boolean canChildBeDismissed() { |
| return true; |
| } |
| |
| @Override |
| public float getMinAllowScrollDistance() { |
| return sScrollSlop; |
| } |
| |
| private void startDestroyAnimation() { |
| final int start = getHeight(); |
| final int end = 0; |
| mAnimatedHeight = start; |
| final ObjectAnimator heightAnimator = |
| ObjectAnimator.ofInt(this, "animatedHeight", start, end); |
| heightAnimator.setInterpolator(new DecelerateInterpolator(2.0f)); |
| heightAnimator.setDuration(sShrinkAnimationDuration); |
| heightAnimator.addListener(new AnimatorListener() { |
| @Override |
| public void onAnimationStart(final Animator animation) { |
| // Do nothing |
| } |
| |
| @Override |
| public void onAnimationRepeat(final Animator animation) { |
| // Do nothing |
| } |
| |
| @Override |
| public void onAnimationEnd(final Animator animation) { |
| // We should no longer exist, so notify the adapter |
| mAdapter.notifyDataSetChanged(); |
| } |
| |
| @Override |
| public void onAnimationCancel(final Animator animation) { |
| // Do nothing |
| } |
| }); |
| heightAnimator.start(); |
| } |
| |
| /** |
| * This method is used by the animator. It is explicitly kept in proguard.flags to prevent it |
| * from being removed, inlined, or obfuscated. |
| * Edit ./vendor/unbundled/packages/apps/UnifiedGmail/proguard.flags |
| * In the future, we want to use @Keep |
| */ |
| public void setAnimatedHeight(final int height) { |
| mAnimatedHeight = height; |
| requestLayout(); |
| } |
| |
| @Override |
| protected void onMeasure(final int widthMeasureSpec, final int heightMeasureSpec) { |
| if (Utils.getDisplayListRightEdgeEffect(mTabletDevice, mListCollapsible, |
| mAdapter.getViewMode())) { |
| mTeaserRightEdge.setVisibility(VISIBLE); |
| } else { |
| mTeaserRightEdge.setVisibility(GONE); |
| } |
| |
| if (mAnimatedHeight == -1) { |
| super.onMeasure(widthMeasureSpec, heightMeasureSpec); |
| } else { |
| setMeasuredDimension(MeasureSpec.getSize(widthMeasureSpec), mAnimatedHeight); |
| } |
| } |
| |
| @Override |
| public boolean commitLeaveBehindItem() { |
| // This view has no leave-behind |
| return false; |
| } |
| } |