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