Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2012 Google Inc. |
| 3 | * Licensed to The Android Open Source Project. |
| 4 | * |
| 5 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 6 | * you may not use this file except in compliance with the License. |
| 7 | * You may obtain a copy of the License at |
| 8 | * |
| 9 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | * |
| 11 | * Unless required by applicable law or agreed to in writing, software |
| 12 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 13 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 14 | * See the License for the specific language governing permissions and |
| 15 | * limitations under the License. |
| 16 | */ |
| 17 | |
| 18 | package com.android.mail.ui; |
| 19 | |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 20 | import android.content.ContentResolver; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 21 | import android.content.Context; |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 22 | import android.content.Loader; |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 23 | import android.content.res.Resources; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 24 | import android.database.Cursor; |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 25 | import android.database.DataSetObserver; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 26 | import android.net.Uri; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 27 | import android.os.AsyncTask; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 28 | import android.os.Bundle; |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 29 | import android.os.SystemClock; |
Andrew Sapperstein | 3af481c | 2013-10-30 10:29:38 -0700 | [diff] [blame] | 30 | import android.support.v4.text.BidiFormatter; |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 31 | import android.support.v4.util.ArrayMap; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 32 | import android.text.TextUtils; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 33 | import android.view.LayoutInflater; |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 34 | import android.view.View; |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 35 | import android.view.View.OnLayoutChangeListener; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 36 | import android.view.ViewGroup; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 37 | import android.webkit.ConsoleMessage; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 38 | import android.webkit.CookieManager; |
| 39 | import android.webkit.CookieSyncManager; |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 40 | import android.webkit.JavascriptInterface; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 41 | import android.webkit.WebChromeClient; |
| 42 | import android.webkit.WebSettings; |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 43 | import android.webkit.WebView; |
Andrew Sapperstein | 821fa87 | 2013-08-21 21:57:39 -0700 | [diff] [blame] | 44 | import android.widget.Button; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 45 | |
Tony Mantler | 821e578 | 2014-01-06 15:33:43 -0800 | [diff] [blame] | 46 | import com.android.emailcommon.mail.Address; |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 47 | import com.android.mail.FormattedDateBuilder; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 48 | import com.android.mail.R; |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 49 | import com.android.mail.analytics.Analytics; |
Jin Cao | 72953f2 | 2014-04-15 18:23:37 -0700 | [diff] [blame] | 50 | import com.android.mail.analytics.AnalyticsTimer; |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 51 | import com.android.mail.browse.ConversationContainer; |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 52 | import com.android.mail.browse.ConversationContainer.OverlayPosition; |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 53 | import com.android.mail.browse.ConversationFooterView.ConversationFooterCallbacks; |
Andrew Sapperstein | 8812d3c | 2013-06-04 17:06:41 -0700 | [diff] [blame] | 54 | import com.android.mail.browse.ConversationMessage; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 55 | import com.android.mail.browse.ConversationOverlayItem; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 56 | import com.android.mail.browse.ConversationViewAdapter; |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 57 | import com.android.mail.browse.ConversationViewAdapter.ConversationFooterItem; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 58 | import com.android.mail.browse.ConversationViewAdapter.MessageFooterItem; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 59 | import com.android.mail.browse.ConversationViewAdapter.MessageHeaderItem; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 60 | import com.android.mail.browse.ConversationViewAdapter.SuperCollapsedBlockItem; |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 61 | import com.android.mail.browse.ConversationViewHeader; |
| 62 | import com.android.mail.browse.ConversationWebView; |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 63 | import com.android.mail.browse.InlineAttachmentViewIntentBuilderCreator; |
| 64 | import com.android.mail.browse.InlineAttachmentViewIntentBuilderCreatorHolder; |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 65 | import com.android.mail.browse.MailWebView.ContentSizeChangeListener; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 66 | import com.android.mail.browse.MessageCursor; |
Andrew Sapperstein | 381c322 | 2014-04-20 12:23:57 -0700 | [diff] [blame] | 67 | import com.android.mail.browse.MessageFooterView; |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 68 | import com.android.mail.browse.MessageHeaderView; |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 69 | import com.android.mail.browse.ScrollIndicatorsView; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 70 | import com.android.mail.browse.SuperCollapsedBlock; |
Andy Huang | 0b7ed6f | 2012-07-25 19:23:26 -0700 | [diff] [blame] | 71 | import com.android.mail.browse.WebViewContextMenu; |
Paul Westbrook | c42ad5e | 2013-05-09 16:52:15 -0700 | [diff] [blame] | 72 | import com.android.mail.content.ObjectCursor; |
Andrew Sapperstein | 562c5ba | 2013-10-09 18:31:50 -0700 | [diff] [blame] | 73 | import com.android.mail.print.PrintUtils; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 74 | import com.android.mail.providers.Account; |
| 75 | import com.android.mail.providers.Conversation; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 76 | import com.android.mail.providers.Message; |
Alice Yang | f323c04 | 2013-10-30 00:15:02 -0700 | [diff] [blame] | 77 | import com.android.mail.providers.Settings; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 78 | import com.android.mail.providers.UIProvider; |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 79 | import com.android.mail.ui.ConversationViewState.ExpansionState; |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 80 | import com.android.mail.utils.ConversationViewUtils; |
Paul Westbrook | b334c90 | 2012-06-25 11:42:46 -0700 | [diff] [blame] | 81 | import com.android.mail.utils.LogTag; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 82 | import com.android.mail.utils.LogUtils; |
Andy Huang | 2e9acfe | 2012-03-15 22:39:36 -0700 | [diff] [blame] | 83 | import com.android.mail.utils.Utils; |
Andy Huang | 543e709 | 2013-04-22 11:44:56 -0700 | [diff] [blame] | 84 | import com.google.common.collect.ImmutableList; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 85 | import com.google.common.collect.Lists; |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 86 | import com.google.common.collect.Maps; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 87 | import com.google.common.collect.Sets; |
Andy Huang | 65fe28f | 2012-04-06 18:08:53 -0700 | [diff] [blame] | 88 | |
Scott Kennedy | eb9a4bd | 2012-11-12 10:33:04 -0800 | [diff] [blame] | 89 | import java.util.ArrayList; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 90 | import java.util.List; |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 91 | import java.util.Map; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 92 | import java.util.Set; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 93 | |
| 94 | /** |
| 95 | * The conversation view UI component. |
| 96 | */ |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 97 | public class ConversationViewFragment extends AbstractConversationViewFragment implements |
Andrew Sapperstein | 4ddda2f | 2013-06-10 11:15:38 -0700 | [diff] [blame] | 98 | SuperCollapsedBlock.OnClickListener, OnLayoutChangeListener, |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 99 | MessageHeaderView.MessageHeaderViewCallbacks, MessageFooterView.MessageFooterCallbacks, |
| 100 | WebViewContextMenu.Callbacks, ConversationFooterCallbacks { |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 101 | |
Paul Westbrook | b334c90 | 2012-06-25 11:42:46 -0700 | [diff] [blame] | 102 | private static final String LOG_TAG = LogTag.getLogTag(); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 103 | public static final String LAYOUT_TAG = "ConvLayout"; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 104 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 105 | /** |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 106 | * Difference in the height of the message header whose details have been expanded/collapsed |
| 107 | */ |
| 108 | private int mDiff = 0; |
| 109 | |
| 110 | /** |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 111 | * Default value for {@link #mLoadWaitReason}. Conversation load will happen immediately. |
| 112 | */ |
| 113 | private final int LOAD_NOW = 0; |
| 114 | /** |
| 115 | * Value for {@link #mLoadWaitReason} that means we are offscreen and waiting for the visible |
| 116 | * conversation to finish loading before beginning our load. |
| 117 | * <p> |
| 118 | * When this value is set, the fragment should register with {@link ConversationListCallbacks} |
| 119 | * to know when the visible conversation is loaded. When it is unset, it should unregister. |
| 120 | */ |
| 121 | private final int LOAD_WAIT_FOR_INITIAL_CONVERSATION = 1; |
| 122 | /** |
| 123 | * Value for {@link #mLoadWaitReason} used when a conversation is too heavyweight to load at |
| 124 | * all when not visible (e.g. requires network fetch, or too complex). Conversation load will |
| 125 | * wait until this fragment is visible. |
| 126 | */ |
| 127 | private final int LOAD_WAIT_UNTIL_VISIBLE = 2; |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 128 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 129 | protected ConversationContainer mConversationContainer; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 130 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 131 | protected ConversationWebView mWebView; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 132 | |
Andrew Sapperstein | a7fa9bf | 2014-03-29 13:33:04 -0700 | [diff] [blame] | 133 | private ConversationViewProgressController mProgressController; |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 134 | |
Andrew Sapperstein | 821fa87 | 2013-08-21 21:57:39 -0700 | [diff] [blame] | 135 | private Button mNewMessageBar; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 136 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 137 | protected HtmlConversationTemplates mTemplates; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 138 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 139 | private final MailJsBridge mJsBridge = new MailJsBridge(); |
| 140 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 141 | protected ConversationViewAdapter mAdapter; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 142 | |
Andrew Sapperstein | b1d184d | 2013-08-09 14:14:31 -0700 | [diff] [blame] | 143 | protected boolean mViewsCreated; |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 144 | // True if we attempted to render before the views were laid out |
| 145 | // We will render immediately once layout is done |
| 146 | private boolean mNeedRender; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 147 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 148 | /** |
| 149 | * Temporary string containing the message bodies of the messages within a super-collapsed |
| 150 | * block, for one-time use during block expansion. We cannot easily pass the body HTML |
| 151 | * into JS without problematic escaping, so hold onto it momentarily and signal JS to fetch it |
| 152 | * using {@link MailJsBridge}. |
| 153 | */ |
| 154 | private String mTempBodiesHtml; |
| 155 | |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 156 | private int mMaxAutoLoadMessages; |
| 157 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 158 | protected int mSideMarginPx; |
Andy Huang | 02f9d18 | 2012-11-28 22:38:02 -0800 | [diff] [blame] | 159 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 160 | /** |
| 161 | * If this conversation fragment is not visible, and it's inappropriate to load up front, |
| 162 | * this is the reason we are waiting. This flag should be cleared once it's okay to load |
| 163 | * the conversation. |
| 164 | */ |
| 165 | private int mLoadWaitReason = LOAD_NOW; |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 166 | |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 167 | private boolean mEnableContentReadySignal; |
Andy Huang | 28b7aee | 2012-08-20 20:27:32 -0700 | [diff] [blame] | 168 | |
mindyp | dde3f9f | 2012-09-10 17:35:35 -0700 | [diff] [blame] | 169 | private ContentSizeChangeListener mWebViewSizeChangeListener; |
| 170 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 171 | private float mWebViewYPercent; |
| 172 | |
| 173 | /** |
| 174 | * Has loadData been called on the WebView yet? |
| 175 | */ |
| 176 | private boolean mWebViewLoadedData; |
| 177 | |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 178 | private long mWebViewLoadStartMs; |
| 179 | |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 180 | private final Map<String, String> mMessageTransforms = Maps.newHashMap(); |
| 181 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 182 | private final DataSetObserver mLoadedObserver = new DataSetObserver() { |
| 183 | @Override |
| 184 | public void onChanged() { |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 185 | getHandler().post(new FragmentRunnable("delayedConversationLoad", |
| 186 | ConversationViewFragment.this) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 187 | @Override |
| 188 | public void go() { |
| 189 | LogUtils.d(LOG_TAG, "CVF load observer fired, this=%s", |
| 190 | ConversationViewFragment.this); |
| 191 | handleDelayedConversationLoad(); |
| 192 | } |
| 193 | }); |
| 194 | } |
| 195 | }; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 196 | |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 197 | private final Runnable mOnProgressDismiss = new FragmentRunnable("onProgressDismiss", this) { |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 198 | @Override |
| 199 | public void go() { |
Scott Kennedy | 58192e5 | 2013-05-08 16:35:57 -0700 | [diff] [blame] | 200 | LogUtils.d(LOG_TAG, "onProgressDismiss go() - isUserVisible() = %b", isUserVisible()); |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 201 | if (isUserVisible()) { |
| 202 | onConversationSeen(); |
| 203 | } |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 204 | mWebView.onRenderComplete(); |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 205 | } |
| 206 | }; |
| 207 | |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 208 | private static final boolean DEBUG_DUMP_CONVERSATION_HTML = false; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 209 | private static final boolean DISABLE_OFFSCREEN_LOADING = false; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 210 | private static final boolean DEBUG_DUMP_CURSOR_CONTENTS = false; |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 211 | |
| 212 | private static final String BUNDLE_KEY_WEBVIEW_Y_PERCENT = |
| 213 | ConversationViewFragment.class.getName() + "webview-y-percent"; |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 214 | |
Andrew Sapperstein | 2fd167d | 2014-01-28 10:07:38 -0800 | [diff] [blame] | 215 | private BidiFormatter mBidiFormatter; |
Andrew Sapperstein | 3af481c | 2013-10-30 10:29:38 -0700 | [diff] [blame] | 216 | |
Vikram Aggarwal | 6c51158 | 2012-02-27 10:59:47 -0800 | [diff] [blame] | 217 | /** |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 218 | * Contains a mapping between inline image attachments and their local message id. |
| 219 | */ |
| 220 | private Map<String, String> mUrlToMessageIdMap; |
| 221 | |
| 222 | /** |
Vikram Aggarwal | 6c51158 | 2012-02-27 10:59:47 -0800 | [diff] [blame] | 223 | * Constructor needs to be public to handle orientation changes and activity lifecycle events. |
| 224 | */ |
Paul Westbrook | f0ea484 | 2013-08-13 16:41:18 -0700 | [diff] [blame] | 225 | public ConversationViewFragment() {} |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 226 | |
| 227 | /** |
| 228 | * Creates a new instance of {@link ConversationViewFragment}, initialized |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 229 | * to display a conversation with other parameters inherited/copied from an existing bundle, |
| 230 | * typically one created using {@link #makeBasicArgs}. |
| 231 | */ |
| 232 | public static ConversationViewFragment newInstance(Bundle existingArgs, |
| 233 | Conversation conversation) { |
| 234 | ConversationViewFragment f = new ConversationViewFragment(); |
| 235 | Bundle args = new Bundle(existingArgs); |
| 236 | args.putParcelable(ARG_CONVERSATION, conversation); |
| 237 | f.setArguments(args); |
| 238 | return f; |
| 239 | } |
| 240 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 241 | @Override |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 242 | public void onAccountChanged(Account newAccount, Account oldAccount) { |
| 243 | // if overview mode has changed, re-render completely (no need to also update headers) |
| 244 | if (isOverviewMode(newAccount) != isOverviewMode(oldAccount)) { |
| 245 | setupOverviewMode(); |
| 246 | final MessageCursor c = getMessageCursor(); |
| 247 | if (c != null) { |
| 248 | renderConversation(c); |
| 249 | } else { |
| 250 | // Null cursor means this fragment is either waiting to load or in the middle of |
| 251 | // loading. Either way, a future render will happen anyway, and the new setting |
| 252 | // will take effect when that happens. |
| 253 | } |
| 254 | return; |
| 255 | } |
| 256 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 257 | // settings may have been updated; refresh views that are known to |
| 258 | // depend on settings |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 259 | mAdapter.notifyDataSetChanged(); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 260 | } |
| 261 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 262 | @Override |
| 263 | public void onActivityCreated(Bundle savedInstanceState) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 264 | LogUtils.d(LOG_TAG, "IN CVF.onActivityCreated, this=%s visible=%s", this, isUserVisible()); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 265 | super.onActivityCreated(savedInstanceState); |
Mark Wei | 1abfcaf | 2012-09-27 11:11:07 -0700 | [diff] [blame] | 266 | |
| 267 | if (mActivity == null || mActivity.isFinishing()) { |
| 268 | // Activity is finishing, just bail. |
| 269 | return; |
| 270 | } |
| 271 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 272 | Context context = getContext(); |
| 273 | mTemplates = new HtmlConversationTemplates(context); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 274 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 275 | final FormattedDateBuilder dateBuilder = new FormattedDateBuilder(context); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 276 | |
Paul Westbrook | 8081df4 | 2012-09-10 15:43:36 -0700 | [diff] [blame] | 277 | mAdapter = new ConversationViewAdapter(mActivity, this, |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 278 | getLoaderManager(), this, this, getContactInfoSource(), this, this, |
Andrew Sapperstein | eb58a09 | 2014-06-24 14:03:49 -0700 | [diff] [blame] | 279 | getListController(), this, mAddressCache, dateBuilder, mBidiFormatter); |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 280 | mConversationContainer.setOverlayAdapter(mAdapter); |
| 281 | |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 282 | // set up snap header (the adapter usually does this with the other ones) |
Andrew Sapperstein | 85ea618 | 2013-10-14 18:21:08 -0700 | [diff] [blame] | 283 | mConversationContainer.getSnapHeader().initialize( |
| 284 | this, mAddressCache, this, getContactInfoSource(), |
| 285 | mActivity.getAccountController().getVeiledAddressMatcher()); |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 286 | |
Andrew Sapperstein | 90eccf9 | 2013-08-22 11:53:23 -0700 | [diff] [blame] | 287 | final Resources resources = getResources(); |
| 288 | mMaxAutoLoadMessages = resources.getInteger(R.integer.max_auto_load_messages); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 289 | |
Andrew Sapperstein | 90eccf9 | 2013-08-22 11:53:23 -0700 | [diff] [blame] | 290 | mSideMarginPx = resources.getDimensionPixelOffset( |
Andy Huang | 02f9d18 | 2012-11-28 22:38:02 -0800 | [diff] [blame] | 291 | R.dimen.conversation_message_content_margin_side); |
| 292 | |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 293 | mUrlToMessageIdMap = new ArrayMap<String, String>(); |
| 294 | final InlineAttachmentViewIntentBuilderCreator creator = |
| 295 | InlineAttachmentViewIntentBuilderCreatorHolder. |
| 296 | getInlineAttachmentViewIntentCreator(); |
Andy Huang | 3c6fd44 | 2014-03-24 19:56:46 -0700 | [diff] [blame] | 297 | final WebViewContextMenu contextMenu = new WebViewContextMenu(getActivity(), |
| 298 | creator.createInlineAttachmentViewIntentBuilder(mAccount.getEmailAddress(), |
| 299 | mConversation != null ? mConversation.id : -1)); |
| 300 | contextMenu.setCallbacks(this); |
| 301 | mWebView.setOnCreateContextMenuListener(contextMenu); |
Andy Huang | 0b7ed6f | 2012-07-25 19:23:26 -0700 | [diff] [blame] | 302 | |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 303 | // set this up here instead of onCreateView to ensure the latest Account is loaded |
| 304 | setupOverviewMode(); |
| 305 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 306 | // Defer the call to initLoader with a Handler. |
| 307 | // We want to wait until we know which fragments are present and their final visibility |
| 308 | // states before going off and doing work. This prevents extraneous loading from occurring |
| 309 | // as the ViewPager shifts about before the initial position is set. |
| 310 | // |
| 311 | // e.g. click on item #10 |
| 312 | // ViewPager.setAdapter() actually first loads #0 and #1 under the assumption that #0 is |
| 313 | // the initial primary item |
| 314 | // Then CPC immediately sets the primary item to #10, which tears down #0/#1 and sets up |
| 315 | // #9/#10/#11. |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 316 | getHandler().post(new FragmentRunnable("showConversation", this) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 317 | @Override |
| 318 | public void go() { |
| 319 | showConversation(); |
| 320 | } |
| 321 | }); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 322 | |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 323 | if (mConversation != null && mConversation.conversationBaseUri != null && |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 324 | !Utils.isEmpty(mAccount.accountCookieQueryUri)) { |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 325 | // Set the cookie for this base url |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 326 | new SetCookieTask(getContext(), mConversation.conversationBaseUri.toString(), |
| 327 | mAccount.accountCookieQueryUri).execute(); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 328 | } |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 329 | } |
| 330 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 331 | @Override |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 332 | public void onCreate(Bundle savedState) { |
| 333 | super.onCreate(savedState); |
| 334 | |
Andrew Sapperstein | b1d184d | 2013-08-09 14:14:31 -0700 | [diff] [blame] | 335 | mWebViewClient = createConversationWebViewClient(); |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 336 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 337 | if (savedState != null) { |
| 338 | mWebViewYPercent = savedState.getFloat(BUNDLE_KEY_WEBVIEW_Y_PERCENT); |
| 339 | } |
Andrew Sapperstein | 3af481c | 2013-10-30 10:29:38 -0700 | [diff] [blame] | 340 | |
Andrew Sapperstein | 2fd167d | 2014-01-28 10:07:38 -0800 | [diff] [blame] | 341 | mBidiFormatter = BidiFormatter.getInstance(); |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 342 | } |
| 343 | |
Andrew Sapperstein | b1d184d | 2013-08-09 14:14:31 -0700 | [diff] [blame] | 344 | protected ConversationWebViewClient createConversationWebViewClient() { |
| 345 | return new ConversationWebViewClient(mAccount); |
| 346 | } |
| 347 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 348 | @Override |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 349 | public View onCreateView(LayoutInflater inflater, |
| 350 | ViewGroup container, Bundle savedInstanceState) { |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 351 | |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 352 | View rootView = inflater.inflate(R.layout.conversation_view, container, false); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 353 | mConversationContainer = (ConversationContainer) rootView |
| 354 | .findViewById(R.id.conversation_container); |
Andy Huang | 8f18778 | 2012-11-06 17:49:25 -0800 | [diff] [blame] | 355 | mConversationContainer.setAccountController(this); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 356 | |
Andrew Sapperstein | 85ea618 | 2013-10-14 18:21:08 -0700 | [diff] [blame] | 357 | final ViewGroup topmostOverlay = |
| 358 | (ViewGroup) mConversationContainer.findViewById(R.id.conversation_topmost_overlay); |
| 359 | inflateSnapHeader(topmostOverlay, inflater); |
| 360 | mConversationContainer.setupSnapHeader(); |
| 361 | |
| 362 | setupNewMessageBar(); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 363 | |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 364 | mProgressController = new ConversationViewProgressController(this, getHandler()); |
| 365 | mProgressController.instantiateProgressIndicators(rootView); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 366 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 367 | mWebView = (ConversationWebView) |
| 368 | mConversationContainer.findViewById(R.id.conversation_webview); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 369 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 370 | mWebView.addJavascriptInterface(mJsBridge, "mail"); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 371 | // On JB or newer, we use the 'webkitAnimationStart' DOM event to signal load complete |
| 372 | // Below JB, try to speed up initial render by having the webview do supplemental draws to |
| 373 | // custom a software canvas. |
mindyp | b941fdb | 2012-09-11 08:28:23 -0700 | [diff] [blame] | 374 | // TODO(mindyp): |
| 375 | //PAGE READINESS SIGNAL FOR JELLYBEAN AND NEWER |
| 376 | // Notify the app on 'webkitAnimationStart' of a simple dummy element with a simple no-op |
| 377 | // animation that immediately runs on page load. The app uses this as a signal that the |
| 378 | // content is loaded and ready to draw, since WebView delays firing this event until the |
| 379 | // layers are composited and everything is ready to draw. |
| 380 | // This signal does not seem to be reliable, so just use the old method for now. |
Andy Huang | f7ac83f | 2013-07-15 15:48:31 -0700 | [diff] [blame] | 381 | final boolean isJBOrLater = Utils.isRunningJellybeanOrLater(); |
| 382 | final boolean isUserVisible = isUserVisible(); |
| 383 | mWebView.setUseSoftwareLayer(!isJBOrLater); |
| 384 | mEnableContentReadySignal = isJBOrLater && isUserVisible; |
| 385 | mWebView.onUserVisibilityChanged(isUserVisible); |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 386 | mWebView.setWebViewClient(mWebViewClient); |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 387 | final WebChromeClient wcc = new WebChromeClient() { |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 388 | @Override |
| 389 | public boolean onConsoleMessage(ConsoleMessage consoleMessage) { |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 390 | if (consoleMessage.messageLevel() == ConsoleMessage.MessageLevel.ERROR) { |
| 391 | LogUtils.wtf(LOG_TAG, "JS: %s (%s:%d) f=%s", consoleMessage.message(), |
| 392 | consoleMessage.sourceId(), consoleMessage.lineNumber(), |
| 393 | ConversationViewFragment.this); |
| 394 | } else { |
| 395 | LogUtils.i(LOG_TAG, "JS: %s (%s:%d) f=%s", consoleMessage.message(), |
| 396 | consoleMessage.sourceId(), consoleMessage.lineNumber(), |
| 397 | ConversationViewFragment.this); |
| 398 | } |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 399 | return true; |
| 400 | } |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 401 | }; |
| 402 | mWebView.setWebChromeClient(wcc); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 403 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 404 | final WebSettings settings = mWebView.getSettings(); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 405 | |
Greg Bullock | f50aafa | 2014-03-22 02:17:00 +0100 | [diff] [blame] | 406 | final ScrollIndicatorsView scrollIndicators = |
Greg Bullock | 82d9f63 | 2014-04-29 10:56:12 +0200 | [diff] [blame] | 407 | (ScrollIndicatorsView) rootView.findViewById(R.id.scroll_indicators); |
Greg Bullock | f50aafa | 2014-03-22 02:17:00 +0100 | [diff] [blame] | 408 | scrollIndicators.setSourceView(mWebView); |
Mark Wei | 56d8385 | 2012-09-19 14:28:50 -0700 | [diff] [blame] | 409 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 410 | settings.setJavaScriptEnabled(true); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 411 | |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 412 | ConversationViewUtils.setTextZoom(getResources(), settings); |
Andy Huang | c319b55 | 2012-04-25 19:53:50 -0700 | [diff] [blame] | 413 | |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 414 | mViewsCreated = true; |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 415 | mWebViewLoadedData = false; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 416 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 417 | return rootView; |
| 418 | } |
| 419 | |
Andrew Sapperstein | 85ea618 | 2013-10-14 18:21:08 -0700 | [diff] [blame] | 420 | protected void inflateSnapHeader(ViewGroup topmostOverlay, LayoutInflater inflater) { |
| 421 | inflater.inflate(R.layout.conversation_topmost_overlay_items, topmostOverlay, true); |
| 422 | } |
| 423 | |
| 424 | protected void setupNewMessageBar() { |
| 425 | mNewMessageBar = (Button) mConversationContainer.findViewById( |
| 426 | R.id.new_message_notification_bar); |
| 427 | mNewMessageBar.setOnClickListener(new View.OnClickListener() { |
| 428 | @Override |
| 429 | public void onClick(View v) { |
| 430 | onNewMessageBarClick(); |
| 431 | } |
| 432 | }); |
| 433 | } |
| 434 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 435 | @Override |
Andy Huang | f7ac83f | 2013-07-15 15:48:31 -0700 | [diff] [blame] | 436 | public void onResume() { |
| 437 | super.onResume(); |
| 438 | if (mWebView != null) { |
| 439 | mWebView.onResume(); |
| 440 | } |
| 441 | } |
| 442 | |
| 443 | @Override |
| 444 | public void onPause() { |
| 445 | super.onPause(); |
| 446 | if (mWebView != null) { |
| 447 | mWebView.onPause(); |
| 448 | } |
| 449 | } |
| 450 | |
| 451 | @Override |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 452 | public void onDestroyView() { |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 453 | super.onDestroyView(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 454 | mConversationContainer.setOverlayAdapter(null); |
| 455 | mAdapter = null; |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 456 | resetLoadWaiting(); // be sure to unregister any active load observer |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 457 | mViewsCreated = false; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 458 | } |
| 459 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 460 | @Override |
| 461 | public void onSaveInstanceState(Bundle outState) { |
| 462 | super.onSaveInstanceState(outState); |
| 463 | |
| 464 | outState.putFloat(BUNDLE_KEY_WEBVIEW_Y_PERCENT, calculateScrollYPercent()); |
| 465 | } |
| 466 | |
| 467 | private float calculateScrollYPercent() { |
Paul Westbrook | 1b56a67 | 2013-04-19 01:19:05 -0700 | [diff] [blame] | 468 | final float p; |
| 469 | if (mWebView == null) { |
| 470 | // onCreateView hasn't been called, return 0 as the user hasn't scrolled the view. |
| 471 | return 0; |
| 472 | } |
| 473 | |
| 474 | final int scrollY = mWebView.getScrollY(); |
| 475 | final int viewH = mWebView.getHeight(); |
| 476 | final int webH = (int) (mWebView.getContentHeight() * mWebView.getScale()); |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 477 | |
| 478 | if (webH == 0 || webH <= viewH) { |
| 479 | p = 0; |
| 480 | } else if (scrollY + viewH >= webH) { |
| 481 | // The very bottom is a special case, it acts as a stronger anchor than the scroll top |
| 482 | // at that point. |
| 483 | p = 1.0f; |
| 484 | } else { |
| 485 | p = (float) scrollY / webH; |
| 486 | } |
| 487 | return p; |
| 488 | } |
| 489 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 490 | private void resetLoadWaiting() { |
| 491 | if (mLoadWaitReason == LOAD_WAIT_FOR_INITIAL_CONVERSATION) { |
| 492 | getListController().unregisterConversationLoadedObserver(mLoadedObserver); |
| 493 | } |
| 494 | mLoadWaitReason = LOAD_NOW; |
| 495 | } |
| 496 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 497 | @Override |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 498 | protected void markUnread() { |
Vikram Aggarwal | d82a31f | 2013-02-05 15:03:00 -0800 | [diff] [blame] | 499 | super.markUnread(); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 500 | // Ignore unsafe calls made after a fragment is detached from an activity |
| 501 | final ControllableActivity activity = (ControllableActivity) getActivity(); |
| 502 | if (activity == null) { |
| 503 | LogUtils.w(LOG_TAG, "ignoring markUnread for conv=%s", mConversation.id); |
| 504 | return; |
| 505 | } |
| 506 | |
Andy Huang | 28e31e2 | 2012-07-26 16:33:15 -0700 | [diff] [blame] | 507 | if (mViewState == null) { |
| 508 | LogUtils.i(LOG_TAG, "ignoring markUnread for conv with no view state (%d)", |
| 509 | mConversation.id); |
| 510 | return; |
| 511 | } |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 512 | activity.getConversationUpdater().markConversationMessagesUnread(mConversation, |
Vikram Aggarwal | 4a878b6 | 2012-07-31 15:09:25 -0700 | [diff] [blame] | 513 | mViewState.getUnreadMessageUris(), mViewState.getConversationInfo()); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 514 | } |
| 515 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 516 | @Override |
| 517 | public void onUserVisibleHintChanged() { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 518 | final boolean userVisible = isUserVisible(); |
Scott Kennedy | 58192e5 | 2013-05-08 16:35:57 -0700 | [diff] [blame] | 519 | LogUtils.d(LOG_TAG, "ConversationViewFragment#onUserVisibleHintChanged(), userVisible = %b", |
| 520 | userVisible); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 521 | |
| 522 | if (!userVisible) { |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 523 | mProgressController.dismissLoadingStatus(); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 524 | } else if (mViewsCreated) { |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 525 | String loadTag = null; |
Andy Huang | eeb4a35 | 2013-11-21 10:56:27 -0800 | [diff] [blame] | 526 | final boolean isInitialLoading; |
| 527 | if (mActivity != null) { |
| 528 | isInitialLoading = mActivity.getConversationUpdater() |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 529 | .isInitialConversationLoading(); |
Andy Huang | eeb4a35 | 2013-11-21 10:56:27 -0800 | [diff] [blame] | 530 | } else { |
| 531 | isInitialLoading = true; |
| 532 | } |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 533 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 534 | if (getMessageCursor() != null) { |
| 535 | LogUtils.d(LOG_TAG, "Fragment is now user-visible, onConversationSeen: %s", this); |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 536 | if (!isInitialLoading) { |
| 537 | loadTag = "preloaded"; |
| 538 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 539 | onConversationSeen(); |
| 540 | } else if (isLoadWaiting()) { |
| 541 | LogUtils.d(LOG_TAG, "Fragment is now user-visible, showing conversation: %s", this); |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 542 | if (!isInitialLoading) { |
| 543 | loadTag = "load_deferred"; |
| 544 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 545 | handleDelayedConversationLoad(); |
| 546 | } |
Andy Huang | e6c9fb6 | 2013-11-15 09:56:20 -0800 | [diff] [blame] | 547 | |
| 548 | if (loadTag != null) { |
| 549 | // pager swipes are visibility transitions to 'visible' except during initial |
| 550 | // pager load on A) enter conversation mode B) rotate C) 2-pane conv-mode list-tap |
| 551 | Analytics.getInstance().sendEvent("pager_swipe", loadTag, |
| 552 | getCurrentFolderTypeDesc(), 0); |
| 553 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 554 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 555 | |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 556 | if (mWebView != null) { |
| 557 | mWebView.onUserVisibilityChanged(userVisible); |
| 558 | } |
Andy Huang | f8cf546 | 2012-10-17 18:29:14 -0700 | [diff] [blame] | 559 | } |
| 560 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 561 | /** |
| 562 | * Will either call initLoader now to begin loading, or set {@link #mLoadWaitReason} and do |
| 563 | * nothing (in which case you should later call {@link #handleDelayedConversationLoad()}). |
| 564 | */ |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 565 | private void showConversation() { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 566 | final int reason; |
| 567 | |
| 568 | if (isUserVisible()) { |
| 569 | LogUtils.i(LOG_TAG, |
| 570 | "SHOWCONV: CVF is user-visible, immediately loading conversation (%s)", this); |
| 571 | reason = LOAD_NOW; |
Andy Huang | 243c236 | 2013-03-01 17:50:35 -0800 | [diff] [blame] | 572 | timerMark("CVF.showConversation"); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 573 | } else { |
| 574 | final boolean disableOffscreenLoading = DISABLE_OFFSCREEN_LOADING |
Alice Yang | 0b8c080 | 2013-09-16 14:35:18 -0700 | [diff] [blame] | 575 | || Utils.isLowRamDevice(getContext()) |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 576 | || (mConversation != null && (mConversation.isRemote |
| 577 | || mConversation.getNumMessages() > mMaxAutoLoadMessages)); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 578 | |
| 579 | // When not visible, we should not immediately load if either this conversation is |
| 580 | // too heavyweight, or if the main/initial conversation is busy loading. |
| 581 | if (disableOffscreenLoading) { |
| 582 | reason = LOAD_WAIT_UNTIL_VISIBLE; |
| 583 | LogUtils.i(LOG_TAG, "SHOWCONV: CVF waiting until visible to load (%s)", this); |
| 584 | } else if (getListController().isInitialConversationLoading()) { |
| 585 | reason = LOAD_WAIT_FOR_INITIAL_CONVERSATION; |
| 586 | LogUtils.i(LOG_TAG, "SHOWCONV: CVF waiting for initial to finish (%s)", this); |
| 587 | getListController().registerConversationLoadedObserver(mLoadedObserver); |
| 588 | } else { |
| 589 | LogUtils.i(LOG_TAG, |
| 590 | "SHOWCONV: CVF is not visible, but no reason to wait. loading now. (%s)", |
| 591 | this); |
| 592 | reason = LOAD_NOW; |
| 593 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 594 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 595 | |
| 596 | mLoadWaitReason = reason; |
| 597 | if (mLoadWaitReason == LOAD_NOW) { |
| 598 | startConversationLoad(); |
| 599 | } |
| 600 | } |
| 601 | |
| 602 | private void handleDelayedConversationLoad() { |
| 603 | resetLoadWaiting(); |
| 604 | startConversationLoad(); |
| 605 | } |
| 606 | |
| 607 | private void startConversationLoad() { |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 608 | mWebView.setVisibility(View.VISIBLE); |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 609 | loadContent(); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 610 | // TODO(mindyp): don't show loading status for a previously rendered |
| 611 | // conversation. Ielieve this is better done by making sure don't show loading status |
| 612 | // until XX ms have passed without loading completed. |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 613 | mProgressController.showLoadingStatus(isUserVisible()); |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 614 | } |
| 615 | |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 616 | /** |
| 617 | * Can be overridden in case a subclass needs to load something other than |
| 618 | * the messages of a conversation. |
| 619 | */ |
| 620 | protected void loadContent() { |
| 621 | getLoaderManager().initLoader(MESSAGE_LOADER, Bundle.EMPTY, getMessageLoaderCallbacks()); |
| 622 | } |
| 623 | |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 624 | private void revealConversation() { |
Andy Huang | 243c236 | 2013-03-01 17:50:35 -0800 | [diff] [blame] | 625 | timerMark("revealing conversation"); |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 626 | mProgressController.dismissLoadingStatus(mOnProgressDismiss); |
Jin Cao | 72953f2 | 2014-04-15 18:23:37 -0700 | [diff] [blame] | 627 | if (isUserVisible()) { |
| 628 | AnalyticsTimer.getInstance().logDuration(AnalyticsTimer.OPEN_CONV_VIEW_FROM_LIST, |
| 629 | true /* isDestructive */, "open_conversation", "from_list", null); |
| 630 | } |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 631 | } |
| 632 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 633 | private boolean isLoadWaiting() { |
| 634 | return mLoadWaitReason != LOAD_NOW; |
| 635 | } |
| 636 | |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 637 | private void renderConversation(MessageCursor messageCursor) { |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 638 | final String convHtml = renderMessageBodies(messageCursor, mEnableContentReadySignal); |
Andy Huang | 243c236 | 2013-03-01 17:50:35 -0800 | [diff] [blame] | 639 | timerMark("rendered conversation"); |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 640 | |
| 641 | if (DEBUG_DUMP_CONVERSATION_HTML) { |
| 642 | java.io.FileWriter fw = null; |
| 643 | try { |
Andrew Sapperstein | f1566b1 | 2013-09-19 15:34:26 -0700 | [diff] [blame] | 644 | fw = new java.io.FileWriter(getSdCardFilePath()); |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 645 | fw.write(convHtml); |
| 646 | } catch (java.io.IOException e) { |
| 647 | e.printStackTrace(); |
| 648 | } finally { |
| 649 | if (fw != null) { |
| 650 | try { |
| 651 | fw.close(); |
| 652 | } catch (java.io.IOException e) { |
| 653 | e.printStackTrace(); |
| 654 | } |
| 655 | } |
| 656 | } |
| 657 | } |
| 658 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 659 | // save off existing scroll position before re-rendering |
| 660 | if (mWebViewLoadedData) { |
| 661 | mWebViewYPercent = calculateScrollYPercent(); |
| 662 | } |
| 663 | |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 664 | mWebView.loadDataWithBaseURL(mBaseUri, convHtml, "text/html", "utf-8", null); |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 665 | mWebViewLoadedData = true; |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 666 | mWebViewLoadStartMs = SystemClock.uptimeMillis(); |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 667 | } |
| 668 | |
Andrew Sapperstein | f1566b1 | 2013-09-19 15:34:26 -0700 | [diff] [blame] | 669 | protected String getSdCardFilePath() { |
| 670 | return "/sdcard/conv" + mConversation.id + ".html"; |
| 671 | } |
| 672 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 673 | /** |
| 674 | * Populate the adapter with overlay views (message headers, super-collapsed blocks, a |
| 675 | * conversation header), and return an HTML document with spacer divs inserted for all overlays. |
| 676 | * |
| 677 | */ |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 678 | protected String renderMessageBodies(MessageCursor messageCursor, |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 679 | boolean enableContentReadySignal) { |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 680 | int pos = -1; |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 681 | |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 682 | LogUtils.d(LOG_TAG, "IN renderMessageBodies, fragment=%s", this); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 683 | boolean allowNetworkImages = false; |
| 684 | |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 685 | // TODO: re-use any existing adapter item state (expanded, details expanded, show pics) |
Andy Huang | 28b7aee | 2012-08-20 20:27:32 -0700 | [diff] [blame] | 686 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 687 | // Walk through the cursor and build up an overlay adapter as you go. |
| 688 | // Each overlay has an entry in the adapter for easy scroll handling in the container. |
| 689 | // Items are not necessarily 1:1 in cursor and adapter because of super-collapsed blocks. |
| 690 | // When adding adapter items, also add their heights to help the container later determine |
| 691 | // overlay dimensions. |
| 692 | |
Andy Huang | db620fe | 2012-08-24 15:45:28 -0700 | [diff] [blame] | 693 | // When re-rendering, prevent ConversationContainer from laying out overlays until after |
| 694 | // the new spacers are positioned by WebView. |
| 695 | mConversationContainer.invalidateSpacerGeometry(); |
| 696 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 697 | mAdapter.clear(); |
| 698 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 699 | // re-evaluate the message parts of the view state, since the messages may have changed |
| 700 | // since the previous render |
| 701 | final ConversationViewState prevState = mViewState; |
| 702 | mViewState = new ConversationViewState(prevState); |
| 703 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 704 | // N.B. the units of height for spacers are actually dp and not px because WebView assumes |
Andy Huang | 2e9acfe | 2012-03-15 22:39:36 -0700 | [diff] [blame] | 705 | // a pixel is an mdpi pixel, unless you set device-dpi. |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 706 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 707 | // add a single conversation header item |
| 708 | final int convHeaderPos = mAdapter.addConversationHeader(mConversation); |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 709 | final int convHeaderPx = measureOverlayHeight(convHeaderPos); |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 710 | |
Andy Huang | 4dc7323 | 2014-02-04 19:58:57 -0800 | [diff] [blame] | 711 | mTemplates.startConversation(mWebView.getViewportWidth(), |
| 712 | mWebView.screenPxToWebPx(mSideMarginPx), mWebView.screenPxToWebPx(convHeaderPx)); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 713 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 714 | int collapsedStart = -1; |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 715 | ConversationMessage prevCollapsedMsg = null; |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 716 | |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 717 | final boolean alwaysShowImages = shouldAlwaysShowImages(); |
Alice Yang | f323c04 | 2013-10-30 00:15:02 -0700 | [diff] [blame] | 718 | |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 719 | boolean prevSafeForImages = alwaysShowImages; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 720 | |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 721 | boolean hasDraft = false; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 722 | while (messageCursor.moveToPosition(++pos)) { |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 723 | final ConversationMessage msg = messageCursor.getMessage(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 724 | |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 725 | final boolean safeForImages = alwaysShowImages || |
Scott Kennedy | 2027384 | 2012-11-07 11:16:21 -0800 | [diff] [blame] | 726 | msg.alwaysShowImages || prevState.getShouldShowImages(msg); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 727 | allowNetworkImages |= safeForImages; |
Andy Huang | 2405528 | 2012-03-27 17:37:06 -0700 | [diff] [blame] | 728 | |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 729 | final Integer savedExpanded = prevState.getExpansionState(msg); |
| 730 | final int expandedState; |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 731 | if (savedExpanded != null) { |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 732 | if (ExpansionState.isSuperCollapsed(savedExpanded) && messageCursor.isLast()) { |
| 733 | // override saved state when this is now the new last message |
| 734 | // this happens to the second-to-last message when you discard a draft |
| 735 | expandedState = ExpansionState.EXPANDED; |
| 736 | } else { |
| 737 | expandedState = savedExpanded; |
| 738 | } |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 739 | } else { |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 740 | // new messages that are not expanded default to being eligible for super-collapse |
Andrew Sapperstein | 605dcfc | 2014-07-03 14:43:35 -0700 | [diff] [blame] | 741 | if (!msg.read || messageCursor.isLast()) { |
| 742 | expandedState = ExpansionState.EXPANDED; |
| 743 | } else if (messageCursor.isFirst()) { |
| 744 | expandedState = ExpansionState.COLLAPSED; |
| 745 | } else { |
| 746 | expandedState = ExpansionState.SUPER_COLLAPSED; |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 747 | hasDraft |= msg.isDraft(); |
Andrew Sapperstein | 605dcfc | 2014-07-03 14:43:35 -0700 | [diff] [blame] | 748 | } |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 749 | } |
Scott Kennedy | 2027384 | 2012-11-07 11:16:21 -0800 | [diff] [blame] | 750 | mViewState.setShouldShowImages(msg, prevState.getShouldShowImages(msg)); |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 751 | mViewState.setExpansionState(msg, expandedState); |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 752 | |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 753 | // save off "read" state from the cursor |
| 754 | // later, the view may not match the cursor (e.g. conversation marked read on open) |
Andy Huang | 423bea2 | 2012-08-21 12:00:49 -0700 | [diff] [blame] | 755 | // however, if a previous state indicated this message was unread, trust that instead |
| 756 | // so "mark unread" marks all originally unread messages |
| 757 | mViewState.setReadState(msg, msg.read && !prevState.isUnread(msg)); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 758 | |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 759 | // We only want to consider this for inclusion in the super collapsed block if |
| 760 | // 1) The we don't have previous state about this message (The first time that the |
| 761 | // user opens a conversation) |
| 762 | // 2) The previously saved state for this message indicates that this message is |
| 763 | // in the super collapsed block. |
| 764 | if (ExpansionState.isSuperCollapsed(expandedState)) { |
| 765 | // contribute to a super-collapsed block that will be emitted just before the |
| 766 | // next expanded header |
| 767 | if (collapsedStart < 0) { |
| 768 | collapsedStart = pos; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 769 | } |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 770 | prevCollapsedMsg = msg; |
| 771 | prevSafeForImages = safeForImages; |
Andrew Sapperstein | 7dc7fa0 | 2013-05-08 16:39:31 -0700 | [diff] [blame] | 772 | |
| 773 | // This line puts the from address in the address cache so that |
| 774 | // we get the sender image for it if it's in a super-collapsed block. |
| 775 | getAddress(msg.getFrom()); |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 776 | continue; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 777 | } |
Andy Huang | 2405528 | 2012-03-27 17:37:06 -0700 | [diff] [blame] | 778 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 779 | // resolve any deferred decisions on previous collapsed items |
| 780 | if (collapsedStart >= 0) { |
| 781 | if (pos - collapsedStart == 1) { |
Andrew Sapperstein | cee3c90 | 2013-07-31 10:52:02 -0700 | [diff] [blame] | 782 | // Special-case for a single collapsed message: no need to super-collapse it. |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 783 | renderMessage(prevCollapsedMsg, false /* expanded */, prevSafeForImages); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 784 | } else { |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 785 | renderSuperCollapsedBlock(collapsedStart, pos - 1, hasDraft); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 786 | } |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 787 | hasDraft = false; // reset hasDraft |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 788 | prevCollapsedMsg = null; |
| 789 | collapsedStart = -1; |
| 790 | } |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 791 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 792 | renderMessage(msg, ExpansionState.isExpanded(expandedState), safeForImages); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 793 | } |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 794 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 795 | final MessageHeaderItem lastHeaderItem = getLastMessageHeaderItem(); |
| 796 | final int convFooterPos = mAdapter.addConversationFooter(lastHeaderItem); |
| 797 | final int convFooterPx = measureOverlayHeight(convFooterPos); |
| 798 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 799 | mWebView.getSettings().setBlockNetworkImage(!allowNetworkImages); |
| 800 | |
Andrew Sapperstein | 2fc6730 | 2013-04-29 18:24:56 -0700 | [diff] [blame] | 801 | final boolean applyTransforms = shouldApplyTransforms(); |
Andy Huang | 57f354c | 2013-04-11 17:23:40 -0700 | [diff] [blame] | 802 | |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 803 | // If the conversation has specified a base uri, use it here, otherwise use mBaseUri |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 804 | return mTemplates.endConversation(convFooterPx, mBaseUri, |
| 805 | mConversation.getBaseUri(mBaseUri), |
Andy Huang | 2160d53 | 2014-02-10 15:43:14 -0800 | [diff] [blame] | 806 | mWebView.getViewportWidth(), mWebView.getWidthInDp(mSideMarginPx), |
| 807 | enableContentReadySignal, isOverviewMode(mAccount), applyTransforms, |
| 808 | applyTransforms); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 809 | } |
Mindy Pereira | 674afa4 | 2012-02-17 14:05:24 -0800 | [diff] [blame] | 810 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 811 | private MessageHeaderItem getLastMessageHeaderItem() { |
| 812 | final int count = mAdapter.getCount(); |
| 813 | if (count < 3) { |
| 814 | LogUtils.wtf(LOG_TAG, "not enough items in the adapter. count: %s", count); |
| 815 | return null; |
| 816 | } |
| 817 | return (MessageHeaderItem) mAdapter.getItem(count - 2); |
| 818 | } |
| 819 | |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 820 | private void renderSuperCollapsedBlock(int start, int end, boolean hasDraft) { |
| 821 | final int blockPos = mAdapter.addSuperCollapsedBlock(start, end, hasDraft); |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 822 | final int blockPx = measureOverlayHeight(blockPos); |
| 823 | mTemplates.appendSuperCollapsedHtml(start, mWebView.screenPxToWebPx(blockPx)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 824 | } |
| 825 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 826 | private void renderMessage(ConversationMessage msg, boolean expanded, boolean safeForImages) { |
Andrew Sapperstein | 14f9374 | 2013-07-25 14:29:56 -0700 | [diff] [blame] | 827 | |
Scott Kennedy | 2027384 | 2012-11-07 11:16:21 -0800 | [diff] [blame] | 828 | final int headerPos = mAdapter.addMessageHeader(msg, expanded, |
| 829 | mViewState.getShouldShowImages(msg)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 830 | final MessageHeaderItem headerItem = (MessageHeaderItem) mAdapter.getItem(headerPos); |
| 831 | |
| 832 | final int footerPos = mAdapter.addMessageFooter(headerItem); |
| 833 | |
| 834 | // Measure item header and footer heights to allocate spacers in HTML |
| 835 | // But since the views themselves don't exist yet, render each item temporarily into |
| 836 | // a host view for measurement. |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 837 | final int headerPx = measureOverlayHeight(headerPos); |
| 838 | final int footerPx = measureOverlayHeight(footerPos); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 839 | |
Andy Huang | 256b35c | 2012-08-22 15:19:13 -0700 | [diff] [blame] | 840 | mTemplates.appendMessageHtml(msg, expanded, safeForImages, |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 841 | mWebView.screenPxToWebPx(headerPx), mWebView.screenPxToWebPx(footerPx)); |
Andy Huang | 243c236 | 2013-03-01 17:50:35 -0800 | [diff] [blame] | 842 | timerMark("rendered message"); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 843 | } |
| 844 | |
| 845 | private String renderCollapsedHeaders(MessageCursor cursor, |
| 846 | SuperCollapsedBlockItem blockToReplace) { |
| 847 | final List<ConversationOverlayItem> replacements = Lists.newArrayList(); |
| 848 | |
| 849 | mTemplates.reset(); |
| 850 | |
Alice Yang | f323c04 | 2013-10-30 00:15:02 -0700 | [diff] [blame] | 851 | final boolean alwaysShowImages = (mAccount != null) && |
| 852 | (mAccount.settings.showImages == Settings.ShowImages.ALWAYS); |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 853 | |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 854 | // In devices with non-integral density multiplier, screen pixels translate to non-integral |
| 855 | // web pixels. Keep track of the error that occurs when we cast all heights to int |
| 856 | float error = 0f; |
Andrew Sapperstein | 14f9374 | 2013-07-25 14:29:56 -0700 | [diff] [blame] | 857 | boolean first = true; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 858 | for (int i = blockToReplace.getStart(), end = blockToReplace.getEnd(); i <= end; i++) { |
| 859 | cursor.moveToPosition(i); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 860 | final ConversationMessage msg = cursor.getMessage(); |
Andrew Sapperstein | 14f9374 | 2013-07-25 14:29:56 -0700 | [diff] [blame] | 861 | |
Andrew Sapperstein | 4ddda2f | 2013-06-10 11:15:38 -0700 | [diff] [blame] | 862 | final MessageHeaderItem header = ConversationViewAdapter.newMessageHeaderItem( |
Andrew Sapperstein | 14f9374 | 2013-07-25 14:29:56 -0700 | [diff] [blame] | 863 | mAdapter, mAdapter.getDateBuilder(), msg, false /* expanded */, |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 864 | alwaysShowImages || mViewState.getShouldShowImages(msg)); |
Andrew Sapperstein | 381c322 | 2014-04-20 12:23:57 -0700 | [diff] [blame] | 865 | final MessageFooterItem footer = mAdapter.newMessageFooterItem(mAdapter, header); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 866 | |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 867 | final int headerPx = measureOverlayHeight(header); |
| 868 | final int footerPx = measureOverlayHeight(footer); |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 869 | error += mWebView.screenPxToWebPxError(headerPx) |
Andrew Sapperstein | 59ccec3 | 2014-06-18 17:22:49 -0700 | [diff] [blame] | 870 | + mWebView.screenPxToWebPxError(footerPx); |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 871 | |
| 872 | // When the error becomes greater than 1 pixel, make the next header 1 pixel taller |
| 873 | int correction = 0; |
| 874 | if (error >= 1) { |
| 875 | correction = 1; |
| 876 | error -= 1; |
| 877 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 878 | |
Andrew Sapperstein | e822148 | 2013-10-02 18:14:58 -0700 | [diff] [blame] | 879 | mTemplates.appendMessageHtml(msg, false /* expanded */, |
| 880 | alwaysShowImages || msg.alwaysShowImages, |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 881 | mWebView.screenPxToWebPx(headerPx) + correction, |
| 882 | mWebView.screenPxToWebPx(footerPx)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 883 | replacements.add(header); |
| 884 | replacements.add(footer); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 885 | |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 886 | mViewState.setExpansionState(msg, ExpansionState.COLLAPSED); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 887 | } |
| 888 | |
| 889 | mAdapter.replaceSuperCollapsedBlock(blockToReplace, replacements); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 890 | mAdapter.notifyDataSetChanged(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 891 | |
| 892 | return mTemplates.emit(); |
| 893 | } |
| 894 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 895 | protected int measureOverlayHeight(int position) { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 896 | return measureOverlayHeight(mAdapter.getItem(position)); |
| 897 | } |
| 898 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 899 | /** |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 900 | * Measure the height of an adapter view by rendering an adapter item into a temporary |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 901 | * host view, and asking the view to immediately measure itself. This method will reuse |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 902 | * a previous adapter view from {@link ConversationContainer}'s scrap views if one was generated |
| 903 | * earlier. |
| 904 | * <p> |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 905 | * After measuring the height, this method also saves the height in the |
| 906 | * {@link ConversationOverlayItem} for later use in overlay positioning. |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 907 | * |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 908 | * @param convItem adapter item with data to render and measure |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 909 | * @return height of the rendered view in screen px |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 910 | */ |
Andrew Sapperstein | afaab17 | 2014-08-07 18:41:15 -0700 | [diff] [blame^] | 911 | private int measureOverlayHeight(ConversationOverlayItem convItem) { |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 912 | final int type = convItem.getType(); |
| 913 | |
| 914 | final View convertView = mConversationContainer.getScrapView(type); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 915 | final View hostView = mAdapter.getView(convItem, convertView, mConversationContainer, |
| 916 | true /* measureOnly */); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 917 | if (convertView == null) { |
| 918 | mConversationContainer.addScrapView(type, hostView); |
| 919 | } |
| 920 | |
Andy Huang | 9875bb4 | 2012-04-04 20:36:21 -0700 | [diff] [blame] | 921 | final int heightPx = mConversationContainer.measureOverlay(hostView); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 922 | convItem.setHeight(heightPx); |
Andy Huang | 9875bb4 | 2012-04-04 20:36:21 -0700 | [diff] [blame] | 923 | convItem.markMeasurementValid(); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 924 | |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 925 | return heightPx; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 926 | } |
| 927 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 928 | @Override |
| 929 | public void onConversationViewHeaderHeightChange(int newHeight) { |
Mark Wei | ab2d998 | 2012-09-25 13:06:17 -0700 | [diff] [blame] | 930 | final int h = mWebView.screenPxToWebPx(newHeight); |
| 931 | |
| 932 | mWebView.loadUrl(String.format("javascript:setConversationHeaderSpacerHeight(%s);", h)); |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 933 | } |
| 934 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 935 | // END conversation header callbacks |
| 936 | |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 937 | // START conversation footer callbacks |
| 938 | |
| 939 | @Override |
| 940 | public void onConversationFooterHeightChange(int newHeight) { |
| 941 | final int h = mWebView.screenPxToWebPx(newHeight); |
| 942 | |
| 943 | mWebView.loadUrl(String.format("javascript:setConversationFooterSpacerHeight(%s);", h)); |
| 944 | } |
| 945 | |
| 946 | // END conversation footer callbacks |
| 947 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 948 | // START message header callbacks |
| 949 | @Override |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 950 | public void setMessageSpacerHeight(MessageHeaderItem item, int newSpacerHeightPx) { |
| 951 | mConversationContainer.invalidateSpacerGeometry(); |
| 952 | |
| 953 | // update message HTML spacer height |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 954 | final int h = mWebView.screenPxToWebPx(newSpacerHeightPx); |
| 955 | LogUtils.i(LAYOUT_TAG, "setting HTML spacer h=%dwebPx (%dscreenPx)", h, |
| 956 | newSpacerHeightPx); |
Vikram Aggarwal | 5349ce1 | 2012-09-24 14:12:40 -0700 | [diff] [blame] | 957 | mWebView.loadUrl(String.format("javascript:setMessageHeaderSpacerHeight('%s', %s);", |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 958 | mTemplates.getMessageDomId(item.getMessage()), h)); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 959 | } |
| 960 | |
| 961 | @Override |
Andrew Sapperstein | 59ccec3 | 2014-06-18 17:22:49 -0700 | [diff] [blame] | 962 | public void setMessageExpanded(MessageHeaderItem item, int newSpacerHeightPx) { |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 963 | mConversationContainer.invalidateSpacerGeometry(); |
| 964 | |
| 965 | // show/hide the HTML message body and update the spacer height |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 966 | final int h = mWebView.screenPxToWebPx(newSpacerHeightPx); |
| 967 | LogUtils.i(LAYOUT_TAG, "setting HTML spacer expanded=%s h=%dwebPx (%dscreenPx)", |
| 968 | item.isExpanded(), h, newSpacerHeightPx); |
Andrew Sapperstein | 59ccec3 | 2014-06-18 17:22:49 -0700 | [diff] [blame] | 969 | mWebView.loadUrl(String.format("javascript:setMessageBodyVisible('%s', %s, %s);", |
| 970 | mTemplates.getMessageDomId(item.getMessage()), item.isExpanded(), h)); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 971 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 972 | mViewState.setExpansionState(item.getMessage(), |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 973 | item.isExpanded() ? ExpansionState.EXPANDED : ExpansionState.COLLAPSED); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 974 | } |
| 975 | |
| 976 | @Override |
Scott Kennedy | eb9a4bd | 2012-11-12 10:33:04 -0800 | [diff] [blame] | 977 | public void showExternalResources(final Message msg) { |
Scott Kennedy | 2027384 | 2012-11-07 11:16:21 -0800 | [diff] [blame] | 978 | mViewState.setShouldShowImages(msg, true); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 979 | mWebView.getSettings().setBlockNetworkImage(false); |
Scott Kennedy | eb9a4bd | 2012-11-12 10:33:04 -0800 | [diff] [blame] | 980 | mWebView.loadUrl("javascript:unblockImages(['" + mTemplates.getMessageDomId(msg) + "']);"); |
| 981 | } |
| 982 | |
| 983 | @Override |
| 984 | public void showExternalResources(final String senderRawAddress) { |
| 985 | mWebView.getSettings().setBlockNetworkImage(false); |
| 986 | |
| 987 | final Address sender = getAddress(senderRawAddress); |
| 988 | final MessageCursor cursor = getMessageCursor(); |
| 989 | |
| 990 | final List<String> messageDomIds = new ArrayList<String>(); |
| 991 | |
| 992 | int pos = -1; |
| 993 | while (cursor.moveToPosition(++pos)) { |
| 994 | final ConversationMessage message = cursor.getMessage(); |
| 995 | if (sender.equals(getAddress(message.getFrom()))) { |
| 996 | message.alwaysShowImages = true; |
| 997 | |
| 998 | mViewState.setShouldShowImages(message, true); |
| 999 | messageDomIds.add(mTemplates.getMessageDomId(message)); |
| 1000 | } |
| 1001 | } |
| 1002 | |
| 1003 | final String url = String.format( |
| 1004 | "javascript:unblockImages(['%s']);", TextUtils.join("','", messageDomIds)); |
| 1005 | mWebView.loadUrl(url); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 1006 | } |
Alice Yang | 1ebc2db | 2013-03-14 21:21:44 -0700 | [diff] [blame] | 1007 | |
| 1008 | @Override |
Andy Huang | 75b52a5 | 2013-03-15 15:40:24 -0700 | [diff] [blame] | 1009 | public boolean supportsMessageTransforms() { |
| 1010 | return true; |
| 1011 | } |
| 1012 | |
| 1013 | @Override |
Alice Yang | 1ebc2db | 2013-03-14 21:21:44 -0700 | [diff] [blame] | 1014 | public String getMessageTransforms(final Message msg) { |
| 1015 | final String domId = mTemplates.getMessageDomId(msg); |
| 1016 | return (domId == null) ? null : mMessageTransforms.get(domId); |
| 1017 | } |
| 1018 | |
James Lemieux | 8e1ffbf | 2014-04-22 15:53:31 -0700 | [diff] [blame] | 1019 | @Override |
| 1020 | public boolean isSecure() { |
| 1021 | return false; |
| 1022 | } |
| 1023 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 1024 | // END message header callbacks |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 1025 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1026 | @Override |
Andrew Sapperstein | 2fc6730 | 2013-04-29 18:24:56 -0700 | [diff] [blame] | 1027 | public void showUntransformedConversation() { |
| 1028 | super.showUntransformedConversation(); |
| 1029 | renderConversation(getMessageCursor()); |
| 1030 | } |
| 1031 | |
| 1032 | @Override |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1033 | public void onSuperCollapsedClick(SuperCollapsedBlockItem item) { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1034 | MessageCursor cursor = getMessageCursor(); |
| 1035 | if (cursor == null || !mViewsCreated) { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1036 | return; |
| 1037 | } |
| 1038 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1039 | mTempBodiesHtml = renderCollapsedHeaders(cursor, item); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1040 | mWebView.loadUrl("javascript:replaceSuperCollapsedBlock(" + item.getStart() + ")"); |
| 1041 | } |
| 1042 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1043 | private void showNewMessageNotification(NewMessagesInfo info) { |
Andrew Sapperstein | 821fa87 | 2013-08-21 21:57:39 -0700 | [diff] [blame] | 1044 | mNewMessageBar.setText(info.getNotificationText()); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1045 | mNewMessageBar.setVisibility(View.VISIBLE); |
| 1046 | } |
| 1047 | |
| 1048 | private void onNewMessageBarClick() { |
| 1049 | mNewMessageBar.setVisibility(View.GONE); |
| 1050 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1051 | renderConversation(getMessageCursor()); // mCursor is already up-to-date |
| 1052 | // per onLoadFinished() |
Andy Huang | 5fbda02 | 2012-02-28 18:22:03 -0800 | [diff] [blame] | 1053 | } |
| 1054 | |
Andrew Sapperstein | e2b2eb7 | 2014-05-13 10:25:34 -0700 | [diff] [blame] | 1055 | private static OverlayPosition[] parsePositions(final int[] topArray, final int[] bottomArray) { |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1056 | final int len = topArray.length; |
| 1057 | final OverlayPosition[] positions = new OverlayPosition[len]; |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 1058 | for (int i = 0; i < len; i++) { |
Andrew Sapperstein | e2b2eb7 | 2014-05-13 10:25:34 -0700 | [diff] [blame] | 1059 | positions[i] = new OverlayPosition(topArray[i], bottomArray[i]); |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 1060 | } |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1061 | return positions; |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 1062 | } |
| 1063 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 1064 | protected Address getAddress(String rawFrom) { |
Paul Westbrook | 0dfae69 | 2013-10-02 00:51:29 -0700 | [diff] [blame] | 1065 | return Utils.getAddress(mAddressCache, rawFrom); |
Andy Huang | 1617481 | 2012-08-16 16:40:35 -0700 | [diff] [blame] | 1066 | } |
| 1067 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1068 | private void ensureContentSizeChangeListener() { |
| 1069 | if (mWebViewSizeChangeListener == null) { |
Andy Huang | c1fb9a9 | 2013-02-11 13:09:12 -0800 | [diff] [blame] | 1070 | mWebViewSizeChangeListener = new ContentSizeChangeListener() { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1071 | @Override |
| 1072 | public void onHeightChange(int h) { |
| 1073 | // When WebKit says the DOM height has changed, re-measure |
| 1074 | // bodies and re-position their headers. |
| 1075 | // This is separate from the typical JavaScript DOM change |
| 1076 | // listeners because cases like NARROW_COLUMNS text reflow do not trigger DOM |
| 1077 | // events. |
| 1078 | mWebView.loadUrl("javascript:measurePositions();"); |
| 1079 | } |
| 1080 | }; |
| 1081 | } |
| 1082 | mWebView.setContentSizeChangeListener(mWebViewSizeChangeListener); |
| 1083 | } |
| 1084 | |
Andrew Sapperstein | 9f957f3 | 2013-07-19 15:18:18 -0700 | [diff] [blame] | 1085 | public static boolean isOverviewMode(Account acct) { |
Andy Huang | ccf6780 | 2013-03-15 14:31:57 -0700 | [diff] [blame] | 1086 | return acct.settings.isOverviewMode(); |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1087 | } |
| 1088 | |
| 1089 | private void setupOverviewMode() { |
Andy Huang | 02f9d18 | 2012-11-28 22:38:02 -0800 | [diff] [blame] | 1090 | // for now, overview mode means use the built-in WebView zoom and disable custom scale |
| 1091 | // gesture handling |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1092 | final boolean overviewMode = isOverviewMode(mAccount); |
| 1093 | final WebSettings settings = mWebView.getSettings(); |
Andy Huang | 4dc7323 | 2014-02-04 19:58:57 -0800 | [diff] [blame] | 1094 | final WebSettings.LayoutAlgorithm layout; |
Andy Huang | 06def56 | 2012-10-14 00:19:11 -0700 | [diff] [blame] | 1095 | settings.setUseWideViewPort(overviewMode); |
Andy Huang | 57f354c | 2013-04-11 17:23:40 -0700 | [diff] [blame] | 1096 | settings.setSupportZoom(overviewMode); |
| 1097 | settings.setBuiltInZoomControls(overviewMode); |
Andy Huang | 4dc7323 | 2014-02-04 19:58:57 -0800 | [diff] [blame] | 1098 | settings.setLoadWithOverviewMode(overviewMode); |
Andy Huang | 57f354c | 2013-04-11 17:23:40 -0700 | [diff] [blame] | 1099 | if (overviewMode) { |
| 1100 | settings.setDisplayZoomControls(false); |
Andy Huang | 4dc7323 | 2014-02-04 19:58:57 -0800 | [diff] [blame] | 1101 | layout = WebSettings.LayoutAlgorithm.NORMAL; |
| 1102 | } else { |
| 1103 | layout = WebSettings.LayoutAlgorithm.NARROW_COLUMNS; |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1104 | } |
Andy Huang | 4dc7323 | 2014-02-04 19:58:57 -0800 | [diff] [blame] | 1105 | settings.setLayoutAlgorithm(layout); |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1106 | } |
| 1107 | |
Andy Huang | 3c6fd44 | 2014-03-24 19:56:46 -0700 | [diff] [blame] | 1108 | @Override |
Andrew Sapperstein | 833123d | 2014-04-23 18:32:44 -0700 | [diff] [blame] | 1109 | public ConversationMessage getMessageForClickedUrl(String url) { |
Andy Huang | 3c6fd44 | 2014-03-24 19:56:46 -0700 | [diff] [blame] | 1110 | final String domMessageId = mUrlToMessageIdMap.get(url); |
| 1111 | if (domMessageId == null) { |
| 1112 | return null; |
| 1113 | } |
| 1114 | final String messageId = mTemplates.getMessageIdForDomId(domMessageId); |
| 1115 | return getMessageCursor().getMessageForId(Long.parseLong(messageId)); |
| 1116 | } |
| 1117 | |
Andrew Sapperstein | b1d184d | 2013-08-09 14:14:31 -0700 | [diff] [blame] | 1118 | public class ConversationWebViewClient extends AbstractConversationWebViewClient { |
Andrew Sapperstein | 4ddda2f | 2013-06-10 11:15:38 -0700 | [diff] [blame] | 1119 | public ConversationWebViewClient(Account account) { |
| 1120 | super(account); |
Andrew Sapperstein | 376294b | 2013-06-06 16:04:26 -0700 | [diff] [blame] | 1121 | } |
| 1122 | |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 1123 | @Override |
| 1124 | public void onPageFinished(WebView view, String url) { |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1125 | // Ignore unsafe calls made after a fragment is detached from an activity. |
| 1126 | // This method needs to, for example, get at the loader manager, which needs |
| 1127 | // the fragment to be added. |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1128 | if (!isAdded() || !mViewsCreated) { |
Paul Westbrook | 006e13c | 2013-07-24 18:40:20 -0700 | [diff] [blame] | 1129 | LogUtils.d(LOG_TAG, "ignoring CVF.onPageFinished, url=%s fragment=%s", url, |
Andy Huang | b95da85 | 2012-07-18 14:16:58 -0700 | [diff] [blame] | 1130 | ConversationViewFragment.this); |
| 1131 | return; |
| 1132 | } |
| 1133 | |
Paul Westbrook | 006e13c | 2013-07-24 18:40:20 -0700 | [diff] [blame] | 1134 | LogUtils.d(LOG_TAG, "IN CVF.onPageFinished, url=%s fragment=%s wv=%s t=%sms", url, |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 1135 | ConversationViewFragment.this, view, |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 1136 | (SystemClock.uptimeMillis() - mWebViewLoadStartMs)); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 1137 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1138 | ensureContentSizeChangeListener(); |
| 1139 | |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1140 | if (!mEnableContentReadySignal) { |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 1141 | revealConversation(); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1142 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1143 | |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1144 | final Set<String> emailAddresses = Sets.newHashSet(); |
Andy Huang | 543e709 | 2013-04-22 11:44:56 -0700 | [diff] [blame] | 1145 | final List<Address> cacheCopy; |
| 1146 | synchronized (mAddressCache) { |
| 1147 | cacheCopy = ImmutableList.copyOf(mAddressCache.values()); |
| 1148 | } |
| 1149 | for (Address addr : cacheCopy) { |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1150 | emailAddresses.add(addr.getAddress()); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1151 | } |
Andrew Sapperstein | 4ddda2f | 2013-06-10 11:15:38 -0700 | [diff] [blame] | 1152 | final ContactLoaderCallbacks callbacks = getContactInfoSource(); |
| 1153 | callbacks.setSenders(emailAddresses); |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1154 | getLoaderManager().restartLoader(CONTACT_LOADER, Bundle.EMPTY, callbacks); |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 1155 | } |
| 1156 | |
Andy Huang | af5d4e0 | 2012-03-19 19:02:12 -0700 | [diff] [blame] | 1157 | @Override |
| 1158 | public boolean shouldOverrideUrlLoading(WebView view, String url) { |
Paul Westbrook | 542fec9 | 2012-09-18 14:47:51 -0700 | [diff] [blame] | 1159 | return mViewsCreated && super.shouldOverrideUrlLoading(view, url); |
Andy Huang | af5d4e0 | 2012-03-19 19:02:12 -0700 | [diff] [blame] | 1160 | } |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 1161 | } |
| 1162 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 1163 | /** |
| 1164 | * NOTE: all public methods must be listed in the proguard flags so that they can be accessed |
| 1165 | * via reflection and not stripped. |
| 1166 | * |
| 1167 | */ |
| 1168 | private class MailJsBridge { |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 1169 | @JavascriptInterface |
Andrew Sapperstein | e2b2eb7 | 2014-05-13 10:25:34 -0700 | [diff] [blame] | 1170 | public void onWebContentGeometryChange(final int[] overlayTopStrs, |
| 1171 | final int[] overlayBottomStrs) { |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 1172 | try { |
| 1173 | getHandler().post(new FragmentRunnable("onWebContentGeometryChange", |
| 1174 | ConversationViewFragment.this) { |
| 1175 | @Override |
| 1176 | public void go() { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1177 | if (!mViewsCreated) { |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 1178 | LogUtils.d(LOG_TAG, "ignoring webContentGeometryChange because views" |
| 1179 | + " are gone, %s", ConversationViewFragment.this); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1180 | return; |
| 1181 | } |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 1182 | mConversationContainer.onGeometryChange( |
| 1183 | parsePositions(overlayTopStrs, overlayBottomStrs)); |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 1184 | if (mDiff != 0) { |
| 1185 | // SCROLL! |
| 1186 | int scale = (int) (mWebView.getScale() / mWebView.getInitialScale()); |
| 1187 | if (scale > 1) { |
| 1188 | mWebView.scrollBy(0, (mDiff * (scale - 1))); |
| 1189 | } |
| 1190 | mDiff = 0; |
| 1191 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1192 | } |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 1193 | }); |
| 1194 | } catch (Throwable t) { |
| 1195 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onWebContentGeometryChange"); |
| 1196 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1197 | } |
| 1198 | |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 1199 | @JavascriptInterface |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1200 | public String getTempMessageBodies() { |
| 1201 | try { |
| 1202 | if (!mViewsCreated) { |
| 1203 | return ""; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 1204 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 1205 | |
| 1206 | final String s = mTempBodiesHtml; |
| 1207 | mTempBodiesHtml = null; |
| 1208 | return s; |
| 1209 | } catch (Throwable t) { |
| 1210 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getTempMessageBodies"); |
| 1211 | return ""; |
| 1212 | } |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 1213 | } |
| 1214 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1215 | @JavascriptInterface |
| 1216 | public String getMessageBody(String domId) { |
| 1217 | try { |
| 1218 | final MessageCursor cursor = getMessageCursor(); |
| 1219 | if (!mViewsCreated || cursor == null) { |
| 1220 | return ""; |
| 1221 | } |
| 1222 | |
| 1223 | int pos = -1; |
| 1224 | while (cursor.moveToPosition(++pos)) { |
| 1225 | final ConversationMessage msg = cursor.getMessage(); |
| 1226 | if (TextUtils.equals(domId, mTemplates.getMessageDomId(msg))) { |
Andy Huang | 986776b | 2014-02-19 18:29:43 -0800 | [diff] [blame] | 1227 | return HtmlConversationTemplates.wrapMessageBody(msg.getBodyAsHtml()); |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1228 | } |
| 1229 | } |
| 1230 | |
| 1231 | return ""; |
| 1232 | |
| 1233 | } catch (Throwable t) { |
| 1234 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getMessageBody"); |
| 1235 | return ""; |
| 1236 | } |
| 1237 | } |
| 1238 | |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 1239 | @JavascriptInterface |
Andy Huang | 543e709 | 2013-04-22 11:44:56 -0700 | [diff] [blame] | 1240 | public String getMessageSender(String domId) { |
| 1241 | try { |
| 1242 | final MessageCursor cursor = getMessageCursor(); |
| 1243 | if (!mViewsCreated || cursor == null) { |
| 1244 | return ""; |
| 1245 | } |
| 1246 | |
| 1247 | int pos = -1; |
| 1248 | while (cursor.moveToPosition(++pos)) { |
| 1249 | final ConversationMessage msg = cursor.getMessage(); |
| 1250 | if (TextUtils.equals(domId, mTemplates.getMessageDomId(msg))) { |
| 1251 | return getAddress(msg.getFrom()).getAddress(); |
| 1252 | } |
| 1253 | } |
| 1254 | |
| 1255 | return ""; |
| 1256 | |
| 1257 | } catch (Throwable t) { |
| 1258 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getMessageSender"); |
| 1259 | return ""; |
| 1260 | } |
| 1261 | } |
| 1262 | |
Andy Huang | 543e709 | 2013-04-22 11:44:56 -0700 | [diff] [blame] | 1263 | @JavascriptInterface |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1264 | public void onContentReady() { |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 1265 | try { |
| 1266 | getHandler().post(new FragmentRunnable("onContentReady", |
| 1267 | ConversationViewFragment.this) { |
| 1268 | @Override |
| 1269 | public void go() { |
| 1270 | try { |
| 1271 | if (mWebViewLoadStartMs != 0) { |
| 1272 | LogUtils.i(LOG_TAG, "IN CVF.onContentReady, f=%s vis=%s t=%sms", |
| 1273 | ConversationViewFragment.this, |
| 1274 | isUserVisible(), |
| 1275 | (SystemClock.uptimeMillis() - mWebViewLoadStartMs)); |
| 1276 | } |
| 1277 | revealConversation(); |
| 1278 | } catch (Throwable t) { |
| 1279 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onContentReady"); |
| 1280 | // Still try to show the conversation. |
| 1281 | revealConversation(); |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 1282 | } |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1283 | } |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 1284 | }); |
| 1285 | } catch (Throwable t) { |
| 1286 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onContentReady"); |
| 1287 | } |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1288 | } |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 1289 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 1290 | @JavascriptInterface |
| 1291 | public float getScrollYPercent() { |
| 1292 | try { |
| 1293 | return mWebViewYPercent; |
| 1294 | } catch (Throwable t) { |
| 1295 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getScrollYPercent"); |
| 1296 | return 0f; |
| 1297 | } |
| 1298 | } |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 1299 | |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 1300 | @JavascriptInterface |
| 1301 | public void onMessageTransform(String messageDomId, String transformText) { |
Andrew Sapperstein | ae92e15 | 2013-05-03 13:55:18 -0700 | [diff] [blame] | 1302 | try { |
| 1303 | LogUtils.i(LOG_TAG, "TRANSFORM: (%s) %s", messageDomId, transformText); |
| 1304 | mMessageTransforms.put(messageDomId, transformText); |
| 1305 | onConversationTransformed(); |
| 1306 | } catch (Throwable t) { |
| 1307 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onMessageTransform"); |
Andrew Sapperstein | 8ec43e8 | 2013-12-17 18:27:55 -0800 | [diff] [blame] | 1308 | } |
| 1309 | } |
| 1310 | |
| 1311 | @JavascriptInterface |
| 1312 | public void onInlineAttachmentsParsed(final String[] urls, final String[] messageIds) { |
| 1313 | try { |
| 1314 | getHandler().post(new FragmentRunnable("onInlineAttachmentsParsed", |
| 1315 | ConversationViewFragment.this) { |
| 1316 | @Override |
| 1317 | public void go() { |
| 1318 | try { |
| 1319 | for (int i = 0, size = urls.length; i < size; i++) { |
| 1320 | mUrlToMessageIdMap.put(urls[i], messageIds[i]); |
| 1321 | } |
| 1322 | } catch (ArrayIndexOutOfBoundsException e) { |
| 1323 | LogUtils.e(LOG_TAG, e, |
| 1324 | "Number of urls does not match number of message ids - %s:%s", |
| 1325 | urls.length, messageIds.length); |
| 1326 | } |
| 1327 | } |
| 1328 | }); |
| 1329 | } catch (Throwable t) { |
| 1330 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onInlineAttachmentsParsed"); |
Andrew Sapperstein | ae92e15 | 2013-05-03 13:55:18 -0700 | [diff] [blame] | 1331 | } |
Andy Huang | 05c70c8 | 2013-03-14 15:15:50 -0700 | [diff] [blame] | 1332 | } |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 1333 | } |
| 1334 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1335 | private class NewMessagesInfo { |
| 1336 | int count; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1337 | int countFromSelf; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1338 | String senderAddress; |
| 1339 | |
| 1340 | /** |
| 1341 | * Return the display text for the new message notification overlay. It will be formatted |
| 1342 | * appropriately for a single new message vs. multiple new messages. |
| 1343 | * |
| 1344 | * @return display text |
| 1345 | */ |
| 1346 | public String getNotificationText() { |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 1347 | Resources res = getResources(); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1348 | if (count > 1) { |
Andrew Sapperstein | 66d6911 | 2013-08-23 12:24:44 -0700 | [diff] [blame] | 1349 | return res.getQuantityString(R.plurals.new_incoming_messages_many, count, count); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1350 | } else { |
Andy Huang | 1617481 | 2012-08-16 16:40:35 -0700 | [diff] [blame] | 1351 | final Address addr = getAddress(senderAddress); |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 1352 | return res.getString(R.string.new_incoming_messages_one, |
Andrew Sapperstein | 2fd167d | 2014-01-28 10:07:38 -0800 | [diff] [blame] | 1353 | mBidiFormatter.unicodeWrap(TextUtils.isEmpty(addr.getPersonal()) |
| 1354 | ? addr.getAddress() : addr.getPersonal())); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1355 | } |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1356 | } |
| 1357 | } |
| 1358 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1359 | @Override |
Paul Westbrook | c42ad5e | 2013-05-09 16:52:15 -0700 | [diff] [blame] | 1360 | public void onMessageCursorLoadFinished(Loader<ObjectCursor<ConversationMessage>> loader, |
| 1361 | MessageCursor newCursor, MessageCursor oldCursor) { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1362 | /* |
| 1363 | * what kind of changes affect the MessageCursor? 1. new message(s) 2. |
| 1364 | * read/unread state change 3. deleted message, either regular or draft |
| 1365 | * 4. updated message, either from self or from others, updated in |
| 1366 | * content or state or sender 5. star/unstar of message (technically |
| 1367 | * similar to #1) 6. other label change Use MessageCursor.hashCode() to |
| 1368 | * sort out interesting vs. no-op cursor updates. |
| 1369 | */ |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1370 | |
Andy Huang | 233d435 | 2012-10-18 14:00:24 -0700 | [diff] [blame] | 1371 | if (oldCursor != null && !oldCursor.isClosed()) { |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1372 | final NewMessagesInfo info = getNewIncomingMessagesInfo(newCursor); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1373 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1374 | if (info.count > 0) { |
| 1375 | // don't immediately render new incoming messages from other |
| 1376 | // senders |
| 1377 | // (to avoid a new message from losing the user's focus) |
| 1378 | LogUtils.i(LOG_TAG, "CONV RENDER: conversation updated" |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1379 | + ", holding cursor for new incoming message (%s)", this); |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1380 | showNewMessageNotification(info); |
| 1381 | return; |
| 1382 | } |
| 1383 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1384 | final int oldState = oldCursor.getStateHashCode(); |
| 1385 | final boolean changed = newCursor.getStateHashCode() != oldState; |
Andy Huang | 233d435 | 2012-10-18 14:00:24 -0700 | [diff] [blame] | 1386 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1387 | if (!changed) { |
| 1388 | final boolean processedInPlace = processInPlaceUpdates(newCursor, oldCursor); |
| 1389 | if (processedInPlace) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1390 | LogUtils.i(LOG_TAG, "CONV RENDER: processed update(s) in place (%s)", this); |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 1391 | } else { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1392 | LogUtils.i(LOG_TAG, "CONV RENDER: uninteresting update" |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1393 | + ", ignoring this conversation update (%s)", this); |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 1394 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1395 | return; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1396 | } else if (info.countFromSelf == 1) { |
| 1397 | // Special-case the very common case of a new cursor that is the same as the old |
| 1398 | // one, except that there is a new message from yourself. This happens upon send. |
| 1399 | final boolean sameExceptNewLast = newCursor.getStateHashCode(1) == oldState; |
| 1400 | if (sameExceptNewLast) { |
| 1401 | LogUtils.i(LOG_TAG, "CONV RENDER: update is a single new message from self" |
| 1402 | + " (%s)", this); |
| 1403 | newCursor.moveToLast(); |
| 1404 | processNewOutgoingMessage(newCursor.getMessage()); |
| 1405 | return; |
| 1406 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1407 | } |
Andy Huang | 6766b6e | 2012-09-28 12:43:52 -0700 | [diff] [blame] | 1408 | // cursors are different, and not due to an incoming message. fall |
| 1409 | // through and render. |
| 1410 | LogUtils.i(LOG_TAG, "CONV RENDER: conversation updated" |
| 1411 | + ", but not due to incoming message. rendering. (%s)", this); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1412 | |
| 1413 | if (DEBUG_DUMP_CURSOR_CONTENTS) { |
| 1414 | LogUtils.i(LOG_TAG, "old cursor: %s", oldCursor.getDebugDump()); |
| 1415 | LogUtils.i(LOG_TAG, "new cursor: %s", newCursor.getDebugDump()); |
| 1416 | } |
Andy Huang | 6766b6e | 2012-09-28 12:43:52 -0700 | [diff] [blame] | 1417 | } else { |
| 1418 | LogUtils.i(LOG_TAG, "CONV RENDER: initial render. (%s)", this); |
Andy Huang | 243c236 | 2013-03-01 17:50:35 -0800 | [diff] [blame] | 1419 | timerMark("message cursor load finished"); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1420 | } |
| 1421 | |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 1422 | renderContent(newCursor); |
| 1423 | } |
| 1424 | |
| 1425 | protected void renderContent(MessageCursor messageCursor) { |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 1426 | // if layout hasn't happened, delay render |
| 1427 | // This is needed in addition to the showConversation() delay to speed |
| 1428 | // up rotation and restoration. |
| 1429 | if (mConversationContainer.getWidth() == 0) { |
| 1430 | mNeedRender = true; |
| 1431 | mConversationContainer.addOnLayoutChangeListener(this); |
| 1432 | } else { |
Andrew Sapperstein | 606dbd7 | 2013-07-30 19:14:23 -0700 | [diff] [blame] | 1433 | renderConversation(messageCursor); |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 1434 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1435 | } |
| 1436 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1437 | private NewMessagesInfo getNewIncomingMessagesInfo(MessageCursor newCursor) { |
| 1438 | final NewMessagesInfo info = new NewMessagesInfo(); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1439 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1440 | int pos = -1; |
| 1441 | while (newCursor.moveToPosition(++pos)) { |
| 1442 | final Message m = newCursor.getMessage(); |
| 1443 | if (!mViewState.contains(m)) { |
| 1444 | LogUtils.i(LOG_TAG, "conversation diff: found new msg: %s", m.uri); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1445 | |
Scott Kennedy | 8960f0a | 2012-11-07 15:35:50 -0800 | [diff] [blame] | 1446 | final Address from = getAddress(m.getFrom()); |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1447 | // distinguish ours from theirs |
| 1448 | // new messages from the account owner should not trigger a |
| 1449 | // notification |
| 1450 | if (mAccount.ownsFromAddress(from.getAddress())) { |
| 1451 | LogUtils.i(LOG_TAG, "found message from self: %s", m.uri); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1452 | info.countFromSelf++; |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1453 | continue; |
| 1454 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1455 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1456 | info.count++; |
Scott Kennedy | 8960f0a | 2012-11-07 15:35:50 -0800 | [diff] [blame] | 1457 | info.senderAddress = m.getFrom(); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1458 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1459 | } |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1460 | return info; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1461 | } |
| 1462 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1463 | private boolean processInPlaceUpdates(MessageCursor newCursor, MessageCursor oldCursor) { |
| 1464 | final Set<String> idsOfChangedBodies = Sets.newHashSet(); |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1465 | final List<Integer> changedOverlayPositions = Lists.newArrayList(); |
| 1466 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1467 | boolean changed = false; |
| 1468 | |
| 1469 | int pos = 0; |
| 1470 | while (true) { |
| 1471 | if (!newCursor.moveToPosition(pos) || !oldCursor.moveToPosition(pos)) { |
| 1472 | break; |
| 1473 | } |
| 1474 | |
| 1475 | final ConversationMessage newMsg = newCursor.getMessage(); |
| 1476 | final ConversationMessage oldMsg = oldCursor.getMessage(); |
| 1477 | |
Jin Cao | 2ef5f08 | 2014-03-31 14:52:38 -0700 | [diff] [blame] | 1478 | // We are going to update the data in the adapter whenever any input fields change. |
| 1479 | // This ensures that the Message object that ComposeActivity uses will be correctly |
| 1480 | // aligned with the most up-to-date data. |
| 1481 | if (!newMsg.isEqual(oldMsg)) { |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1482 | mAdapter.updateItemsForMessage(newMsg, changedOverlayPositions); |
Jin Cao | 6a2df25 | 2014-05-29 15:12:33 -0700 | [diff] [blame] | 1483 | LogUtils.i(LOG_TAG, "msg #%d (%d): detected field(s) change. sendingState=%s", |
| 1484 | pos, newMsg.id, newMsg.sendingState); |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1485 | } |
| 1486 | |
| 1487 | // update changed message bodies in-place |
| 1488 | if (!TextUtils.equals(newMsg.bodyHtml, oldMsg.bodyHtml) || |
| 1489 | !TextUtils.equals(newMsg.bodyText, oldMsg.bodyText)) { |
| 1490 | // maybe just set a flag to notify JS to re-request changed bodies |
| 1491 | idsOfChangedBodies.add('"' + mTemplates.getMessageDomId(newMsg) + '"'); |
| 1492 | LogUtils.i(LOG_TAG, "msg #%d (%d): detected body change", pos, newMsg.id); |
| 1493 | } |
| 1494 | |
| 1495 | pos++; |
| 1496 | } |
| 1497 | |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1498 | |
| 1499 | if (!changedOverlayPositions.isEmpty()) { |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1500 | // notify once after the entire adapter is updated |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1501 | mConversationContainer.onOverlayModelUpdate(changedOverlayPositions); |
| 1502 | changed = true; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1503 | } |
| 1504 | |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 1505 | final ConversationFooterItem footerItem = mAdapter.getFooterItem(); |
| 1506 | if (footerItem != null) { |
| 1507 | footerItem.invalidateMeasurement(); |
| 1508 | } |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1509 | if (!idsOfChangedBodies.isEmpty()) { |
| 1510 | mWebView.loadUrl(String.format("javascript:replaceMessageBodies([%s]);", |
| 1511 | TextUtils.join(",", idsOfChangedBodies))); |
| 1512 | changed = true; |
| 1513 | } |
| 1514 | |
| 1515 | return changed; |
| 1516 | } |
| 1517 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1518 | private void processNewOutgoingMessage(ConversationMessage msg) { |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 1519 | // Temporarily remove the ConversationFooterItem and its view. |
| 1520 | // It will get re-added right after the new message is added. |
| 1521 | final ConversationFooterItem footerItem = mAdapter.removeFooterItem(); |
| 1522 | mConversationContainer.removeViewAtAdapterIndex(footerItem.getPosition()); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1523 | mTemplates.reset(); |
| 1524 | // this method will add some items to mAdapter, but we deliberately want to avoid notifying |
| 1525 | // adapter listeners (i.e. ConversationContainer) until onWebContentGeometryChange is next |
| 1526 | // called, to prevent N+1 headers rendering with N message bodies. |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 1527 | renderMessage(msg, true /* expanded */, msg.alwaysShowImages); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1528 | mTempBodiesHtml = mTemplates.emit(); |
| 1529 | |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 1530 | if (footerItem != null) { |
| 1531 | footerItem.setLastMessageHeaderItem(getLastMessageHeaderItem()); |
Andrew Sapperstein | 735a22a | 2014-07-11 03:57:42 -0700 | [diff] [blame] | 1532 | footerItem.invalidateMeasurement(); |
Andrew Sapperstein | e2a30e1 | 2014-07-02 22:36:56 -0700 | [diff] [blame] | 1533 | mAdapter.addItem(footerItem); |
| 1534 | } |
| 1535 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1536 | mViewState.setExpansionState(msg, ExpansionState.EXPANDED); |
| 1537 | // FIXME: should the provider set this as initial state? |
| 1538 | mViewState.setReadState(msg, false /* read */); |
| 1539 | |
Andy Huang | 91d782a | 2012-10-25 12:37:29 -0700 | [diff] [blame] | 1540 | // From now until the updated spacer geometry is returned, the adapter items are mismatched |
| 1541 | // with the existing spacers. Do not let them layout. |
| 1542 | mConversationContainer.invalidateSpacerGeometry(); |
| 1543 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1544 | mWebView.loadUrl("javascript:appendMessageHtml();"); |
| 1545 | } |
| 1546 | |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 1547 | private static class SetCookieTask extends AsyncTask<Void, Void, Void> { |
| 1548 | private final Context mContext; |
| 1549 | private final String mUri; |
| 1550 | private final Uri mAccountCookieQueryUri; |
| 1551 | private final ContentResolver mResolver; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1552 | |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 1553 | /* package */ SetCookieTask(Context context, String baseUri, Uri accountCookieQueryUri) { |
| 1554 | mContext = context; |
| 1555 | mUri = baseUri; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1556 | mAccountCookieQueryUri = accountCookieQueryUri; |
| 1557 | mResolver = context.getContentResolver(); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1558 | } |
| 1559 | |
| 1560 | @Override |
| 1561 | public Void doInBackground(Void... args) { |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 1562 | // First query for the cookie string from the UI provider |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1563 | final Cursor cookieCursor = mResolver.query(mAccountCookieQueryUri, |
| 1564 | UIProvider.ACCOUNT_COOKIE_PROJECTION, null, null, null); |
| 1565 | if (cookieCursor == null) { |
| 1566 | return null; |
| 1567 | } |
| 1568 | |
| 1569 | try { |
| 1570 | if (cookieCursor.moveToFirst()) { |
| 1571 | final String cookie = cookieCursor.getString( |
| 1572 | cookieCursor.getColumnIndex(UIProvider.AccountCookieColumns.COOKIE)); |
| 1573 | |
| 1574 | if (cookie != null) { |
| 1575 | final CookieSyncManager csm = |
Andrew Sapperstein | f59d01c | 2014-02-20 10:27:06 -0800 | [diff] [blame] | 1576 | CookieSyncManager.createInstance(mContext); |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1577 | CookieManager.getInstance().setCookie(mUri, cookie); |
| 1578 | csm.sync(); |
| 1579 | } |
| 1580 | } |
| 1581 | |
| 1582 | } finally { |
| 1583 | cookieCursor.close(); |
| 1584 | } |
| 1585 | |
| 1586 | |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1587 | return null; |
| 1588 | } |
| 1589 | } |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1590 | |
mindyp | 26d4d2d | 2012-09-18 17:30:32 -0700 | [diff] [blame] | 1591 | @Override |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1592 | public void onConversationUpdated(Conversation conv) { |
| 1593 | final ConversationViewHeader headerView = (ConversationViewHeader) mConversationContainer |
| 1594 | .findViewById(R.id.conversation_header); |
mindyp | b2b98ba | 2012-09-24 14:13:58 -0700 | [diff] [blame] | 1595 | mConversation = conv; |
mindyp | 9e0b236 | 2012-09-09 16:31:21 -0700 | [diff] [blame] | 1596 | if (headerView != null) { |
| 1597 | headerView.onConversationUpdated(conv); |
| 1598 | } |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1599 | } |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 1600 | |
| 1601 | @Override |
| 1602 | public void onLayoutChange(View v, int left, int top, int right, |
| 1603 | int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) { |
| 1604 | boolean sizeChanged = mNeedRender |
| 1605 | && mConversationContainer.getWidth() != 0; |
| 1606 | if (sizeChanged) { |
| 1607 | mNeedRender = false; |
| 1608 | mConversationContainer.removeOnLayoutChangeListener(this); |
| 1609 | renderConversation(getMessageCursor()); |
| 1610 | } |
| 1611 | } |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 1612 | |
| 1613 | @Override |
James Lemieux | 7cad280 | 2014-01-09 15:00:53 -0800 | [diff] [blame] | 1614 | public void setMessageDetailsExpanded(MessageHeaderItem i, boolean expanded, int heightBefore) { |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 1615 | mDiff = (expanded ? 1 : -1) * Math.abs(i.getHeight() - heightBefore); |
| 1616 | } |
Andy Huang | 02f9d18 | 2012-11-28 22:38:02 -0800 | [diff] [blame] | 1617 | |
James Lemieux | 7cad280 | 2014-01-09 15:00:53 -0800 | [diff] [blame] | 1618 | /** |
| 1619 | * @return {@code true} because either the Print or Print All menu item is shown in GMail |
| 1620 | */ |
| 1621 | @Override |
| 1622 | protected boolean shouldShowPrintInOverflow() { |
| 1623 | return true; |
| 1624 | } |
| 1625 | |
| 1626 | @Override |
Andrew Sapperstein | 5c1692a | 2013-09-16 11:56:13 -0700 | [diff] [blame] | 1627 | protected void printConversation() { |
Andrew Sapperstein | 234d353 | 2013-10-29 14:54:04 -0700 | [diff] [blame] | 1628 | PrintUtils.printConversation(mActivity.getActivityContext(), getMessageCursor(), |
| 1629 | mAddressCache, mConversation.getBaseUri(mBaseUri), true /* useJavascript */); |
Andrew Sapperstein | 5c1692a | 2013-09-16 11:56:13 -0700 | [diff] [blame] | 1630 | } |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 1631 | } |