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 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 20 | |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 21 | import android.content.ContentResolver; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 22 | import android.content.Context; |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 23 | import android.content.Loader; |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 24 | import android.content.res.Resources; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 25 | import android.database.Cursor; |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 26 | import android.database.DataSetObserver; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 27 | import android.net.Uri; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 28 | import android.os.AsyncTask; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 29 | import android.os.Bundle; |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 30 | import android.os.SystemClock; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 31 | import android.text.TextUtils; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 32 | import android.view.LayoutInflater; |
| 33 | import android.view.View; |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 34 | import android.view.View.OnLayoutChangeListener; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 35 | import android.view.ViewGroup; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 36 | import android.webkit.ConsoleMessage; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 37 | import android.webkit.CookieManager; |
| 38 | import android.webkit.CookieSyncManager; |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 39 | import android.webkit.JavascriptInterface; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 40 | import android.webkit.WebChromeClient; |
| 41 | import android.webkit.WebSettings; |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 42 | import android.webkit.WebView; |
| 43 | import android.webkit.WebViewClient; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 44 | import android.widget.TextView; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 45 | |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 46 | import com.android.mail.FormattedDateBuilder; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 47 | import com.android.mail.R; |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 48 | import com.android.mail.browse.ConversationContainer; |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 49 | import com.android.mail.browse.ConversationContainer.OverlayPosition; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 50 | import com.android.mail.browse.ConversationOverlayItem; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 51 | import com.android.mail.browse.ConversationViewAdapter; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 52 | import com.android.mail.browse.ConversationViewAdapter.MessageFooterItem; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 53 | import com.android.mail.browse.ConversationViewAdapter.MessageHeaderItem; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 54 | import com.android.mail.browse.ConversationViewAdapter.SuperCollapsedBlockItem; |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 55 | import com.android.mail.browse.ConversationViewHeader; |
| 56 | import com.android.mail.browse.ConversationWebView; |
mindyp | dde3f9f | 2012-09-10 17:35:35 -0700 | [diff] [blame] | 57 | import com.android.mail.browse.ConversationWebView.ContentSizeChangeListener; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 58 | import com.android.mail.browse.MessageCursor; |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 59 | import com.android.mail.browse.MessageCursor.ConversationController; |
Andy Huang | 28b7aee | 2012-08-20 20:27:32 -0700 | [diff] [blame] | 60 | import com.android.mail.browse.MessageCursor.ConversationMessage; |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 61 | import com.android.mail.browse.MessageHeaderView; |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 62 | import com.android.mail.browse.MessageHeaderView.MessageHeaderViewCallbacks; |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 63 | import com.android.mail.browse.ScrollIndicatorsView; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 64 | import com.android.mail.browse.SuperCollapsedBlock; |
Andy Huang | 0b7ed6f | 2012-07-25 19:23:26 -0700 | [diff] [blame] | 65 | import com.android.mail.browse.WebViewContextMenu; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 66 | import com.android.mail.providers.Account; |
Andy Huang | 65fe28f | 2012-04-06 18:08:53 -0700 | [diff] [blame] | 67 | import com.android.mail.providers.Address; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 68 | import com.android.mail.providers.Conversation; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 69 | import com.android.mail.providers.Message; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 70 | import com.android.mail.providers.UIProvider; |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 71 | import com.android.mail.ui.ConversationViewState.ExpansionState; |
Paul Westbrook | b334c90 | 2012-06-25 11:42:46 -0700 | [diff] [blame] | 72 | import com.android.mail.utils.LogTag; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 73 | import com.android.mail.utils.LogUtils; |
Andy Huang | 2e9acfe | 2012-03-15 22:39:36 -0700 | [diff] [blame] | 74 | import com.android.mail.utils.Utils; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 75 | import com.google.common.collect.Lists; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 76 | import com.google.common.collect.Sets; |
Andy Huang | 65fe28f | 2012-04-06 18:08:53 -0700 | [diff] [blame] | 77 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 78 | import java.util.List; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 79 | import java.util.Set; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 80 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 81 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 82 | /** |
| 83 | * The conversation view UI component. |
| 84 | */ |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 85 | public final class ConversationViewFragment extends AbstractConversationViewFragment implements |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 86 | SuperCollapsedBlock.OnClickListener, |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 87 | OnLayoutChangeListener { |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 88 | |
Paul Westbrook | b334c90 | 2012-06-25 11:42:46 -0700 | [diff] [blame] | 89 | private static final String LOG_TAG = LogTag.getLogTag(); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 90 | public static final String LAYOUT_TAG = "ConvLayout"; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 91 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 92 | /** |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 93 | * Difference in the height of the message header whose details have been expanded/collapsed |
| 94 | */ |
| 95 | private int mDiff = 0; |
| 96 | |
| 97 | /** |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 98 | * Default value for {@link #mLoadWaitReason}. Conversation load will happen immediately. |
| 99 | */ |
| 100 | private final int LOAD_NOW = 0; |
| 101 | /** |
| 102 | * Value for {@link #mLoadWaitReason} that means we are offscreen and waiting for the visible |
| 103 | * conversation to finish loading before beginning our load. |
| 104 | * <p> |
| 105 | * When this value is set, the fragment should register with {@link ConversationListCallbacks} |
| 106 | * to know when the visible conversation is loaded. When it is unset, it should unregister. |
| 107 | */ |
| 108 | private final int LOAD_WAIT_FOR_INITIAL_CONVERSATION = 1; |
| 109 | /** |
| 110 | * Value for {@link #mLoadWaitReason} used when a conversation is too heavyweight to load at |
| 111 | * all when not visible (e.g. requires network fetch, or too complex). Conversation load will |
| 112 | * wait until this fragment is visible. |
| 113 | */ |
| 114 | private final int LOAD_WAIT_UNTIL_VISIBLE = 2; |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 115 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 116 | private ConversationContainer mConversationContainer; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 117 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 118 | private ConversationWebView mWebView; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 119 | |
Mark Wei | 56d8385 | 2012-09-19 14:28:50 -0700 | [diff] [blame] | 120 | private ScrollIndicatorsView mScrollIndicators; |
| 121 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 122 | private View mNewMessageBar; |
| 123 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 124 | private HtmlConversationTemplates mTemplates; |
| 125 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 126 | private final MailJsBridge mJsBridge = new MailJsBridge(); |
| 127 | |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 128 | private final WebViewClient mWebViewClient = new ConversationWebViewClient(); |
| 129 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 130 | private ConversationViewAdapter mAdapter; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 131 | |
| 132 | private boolean mViewsCreated; |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 133 | // True if we attempted to render before the views were laid out |
| 134 | // We will render immediately once layout is done |
| 135 | private boolean mNeedRender; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 136 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 137 | /** |
| 138 | * Temporary string containing the message bodies of the messages within a super-collapsed |
| 139 | * block, for one-time use during block expansion. We cannot easily pass the body HTML |
| 140 | * into JS without problematic escaping, so hold onto it momentarily and signal JS to fetch it |
| 141 | * using {@link MailJsBridge}. |
| 142 | */ |
| 143 | private String mTempBodiesHtml; |
| 144 | |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 145 | private int mMaxAutoLoadMessages; |
| 146 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 147 | /** |
| 148 | * If this conversation fragment is not visible, and it's inappropriate to load up front, |
| 149 | * this is the reason we are waiting. This flag should be cleared once it's okay to load |
| 150 | * the conversation. |
| 151 | */ |
| 152 | private int mLoadWaitReason = LOAD_NOW; |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 153 | |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 154 | private boolean mEnableContentReadySignal; |
Andy Huang | 28b7aee | 2012-08-20 20:27:32 -0700 | [diff] [blame] | 155 | |
mindyp | dde3f9f | 2012-09-10 17:35:35 -0700 | [diff] [blame] | 156 | private ContentSizeChangeListener mWebViewSizeChangeListener; |
| 157 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 158 | private float mWebViewYPercent; |
| 159 | |
| 160 | /** |
| 161 | * Has loadData been called on the WebView yet? |
| 162 | */ |
| 163 | private boolean mWebViewLoadedData; |
| 164 | |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 165 | private long mWebViewLoadStartMs; |
| 166 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 167 | private final DataSetObserver mLoadedObserver = new DataSetObserver() { |
| 168 | @Override |
| 169 | public void onChanged() { |
| 170 | getHandler().post(new FragmentRunnable("delayedConversationLoad") { |
| 171 | @Override |
| 172 | public void go() { |
| 173 | LogUtils.d(LOG_TAG, "CVF load observer fired, this=%s", |
| 174 | ConversationViewFragment.this); |
| 175 | handleDelayedConversationLoad(); |
| 176 | } |
| 177 | }); |
| 178 | } |
| 179 | }; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 180 | |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 181 | private final Runnable mOnProgressDismiss = new FragmentRunnable("onProgressDismiss") { |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 182 | @Override |
| 183 | public void go() { |
| 184 | if (isUserVisible()) { |
| 185 | onConversationSeen(); |
| 186 | } |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 187 | mWebView.onRenderComplete(); |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 188 | } |
| 189 | }; |
| 190 | |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 191 | private static final boolean DEBUG_DUMP_CONVERSATION_HTML = false; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 192 | private static final boolean DISABLE_OFFSCREEN_LOADING = false; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 193 | private static final boolean DEBUG_DUMP_CURSOR_CONTENTS = false; |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 194 | |
| 195 | private static final String BUNDLE_KEY_WEBVIEW_Y_PERCENT = |
| 196 | ConversationViewFragment.class.getName() + "webview-y-percent"; |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 197 | |
Vikram Aggarwal | 6c51158 | 2012-02-27 10:59:47 -0800 | [diff] [blame] | 198 | /** |
| 199 | * Constructor needs to be public to handle orientation changes and activity lifecycle events. |
| 200 | */ |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 201 | public ConversationViewFragment() { |
Vikram Aggarwal | 6c51158 | 2012-02-27 10:59:47 -0800 | [diff] [blame] | 202 | super(); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 203 | } |
| 204 | |
| 205 | /** |
| 206 | * Creates a new instance of {@link ConversationViewFragment}, initialized |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 207 | * to display a conversation with other parameters inherited/copied from an existing bundle, |
| 208 | * typically one created using {@link #makeBasicArgs}. |
| 209 | */ |
| 210 | public static ConversationViewFragment newInstance(Bundle existingArgs, |
| 211 | Conversation conversation) { |
| 212 | ConversationViewFragment f = new ConversationViewFragment(); |
| 213 | Bundle args = new Bundle(existingArgs); |
| 214 | args.putParcelable(ARG_CONVERSATION, conversation); |
| 215 | f.setArguments(args); |
| 216 | return f; |
| 217 | } |
| 218 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 219 | @Override |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 220 | public void onAccountChanged(Account newAccount, Account oldAccount) { |
| 221 | // if overview mode has changed, re-render completely (no need to also update headers) |
| 222 | if (isOverviewMode(newAccount) != isOverviewMode(oldAccount)) { |
| 223 | setupOverviewMode(); |
| 224 | final MessageCursor c = getMessageCursor(); |
| 225 | if (c != null) { |
| 226 | renderConversation(c); |
| 227 | } else { |
| 228 | // Null cursor means this fragment is either waiting to load or in the middle of |
| 229 | // loading. Either way, a future render will happen anyway, and the new setting |
| 230 | // will take effect when that happens. |
| 231 | } |
| 232 | return; |
| 233 | } |
| 234 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 235 | // settings may have been updated; refresh views that are known to |
| 236 | // depend on settings |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 237 | mAdapter.notifyDataSetChanged(); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 238 | } |
| 239 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 240 | @Override |
| 241 | public void onActivityCreated(Bundle savedInstanceState) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 242 | 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] | 243 | super.onActivityCreated(savedInstanceState); |
Mark Wei | 1abfcaf | 2012-09-27 11:11:07 -0700 | [diff] [blame] | 244 | |
| 245 | if (mActivity == null || mActivity.isFinishing()) { |
| 246 | // Activity is finishing, just bail. |
| 247 | return; |
| 248 | } |
| 249 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 250 | Context context = getContext(); |
| 251 | mTemplates = new HtmlConversationTemplates(context); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 252 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 253 | final FormattedDateBuilder dateBuilder = new FormattedDateBuilder(context); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 254 | |
Paul Westbrook | 8081df4 | 2012-09-10 15:43:36 -0700 | [diff] [blame] | 255 | mAdapter = new ConversationViewAdapter(mActivity, this, |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 256 | getLoaderManager(), this, getContactInfoSource(), this, |
Paul Westbrook | 8081df4 | 2012-09-10 15:43:36 -0700 | [diff] [blame] | 257 | this, mAddressCache, dateBuilder); |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 258 | mConversationContainer.setOverlayAdapter(mAdapter); |
| 259 | |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 260 | // set up snap header (the adapter usually does this with the other ones) |
| 261 | final MessageHeaderView snapHeader = mConversationContainer.getSnapHeader(); |
Andy Huang | 28b7aee | 2012-08-20 20:27:32 -0700 | [diff] [blame] | 262 | snapHeader.initialize(dateBuilder, this, mAddressCache); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 263 | snapHeader.setCallbacks(this); |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 264 | snapHeader.setContactInfoSource(getContactInfoSource()); |
Andy Huang | 59e0b18 | 2012-08-14 14:32:23 -0700 | [diff] [blame] | 265 | |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 266 | mMaxAutoLoadMessages = getResources().getInteger(R.integer.max_auto_load_messages); |
| 267 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 268 | mWebView.setOnCreateContextMenuListener(new WebViewContextMenu(getActivity())); |
Andy Huang | 0b7ed6f | 2012-07-25 19:23:26 -0700 | [diff] [blame] | 269 | |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 270 | // set this up here instead of onCreateView to ensure the latest Account is loaded |
| 271 | setupOverviewMode(); |
| 272 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 273 | // Defer the call to initLoader with a Handler. |
| 274 | // We want to wait until we know which fragments are present and their final visibility |
| 275 | // states before going off and doing work. This prevents extraneous loading from occurring |
| 276 | // as the ViewPager shifts about before the initial position is set. |
| 277 | // |
| 278 | // e.g. click on item #10 |
| 279 | // ViewPager.setAdapter() actually first loads #0 and #1 under the assumption that #0 is |
| 280 | // the initial primary item |
| 281 | // Then CPC immediately sets the primary item to #10, which tears down #0/#1 and sets up |
| 282 | // #9/#10/#11. |
| 283 | getHandler().post(new FragmentRunnable("showConversation") { |
| 284 | @Override |
| 285 | public void go() { |
| 286 | showConversation(); |
| 287 | } |
| 288 | }); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 289 | |
| 290 | if (mConversation.conversationBaseUri != null && |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 291 | !Utils.isEmpty(mAccount.accoutCookieQueryUri)) { |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 292 | // Set the cookie for this base url |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 293 | new SetCookieTask(getContext(), mConversation.conversationBaseUri, |
| 294 | mAccount.accoutCookieQueryUri).execute(); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 295 | } |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 296 | } |
| 297 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 298 | @Override |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 299 | public void onCreate(Bundle savedState) { |
| 300 | super.onCreate(savedState); |
| 301 | |
| 302 | if (savedState != null) { |
| 303 | mWebViewYPercent = savedState.getFloat(BUNDLE_KEY_WEBVIEW_Y_PERCENT); |
| 304 | } |
| 305 | } |
| 306 | |
| 307 | @Override |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 308 | public View onCreateView(LayoutInflater inflater, |
| 309 | ViewGroup container, Bundle savedInstanceState) { |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 310 | |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 311 | View rootView = inflater.inflate(R.layout.conversation_view, container, false); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 312 | mConversationContainer = (ConversationContainer) rootView |
| 313 | .findViewById(R.id.conversation_container); |
Andy Huang | 8f18778 | 2012-11-06 17:49:25 -0800 | [diff] [blame] | 314 | mConversationContainer.setAccountController(this); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 315 | |
| 316 | mNewMessageBar = mConversationContainer.findViewById(R.id.new_message_notification_bar); |
| 317 | mNewMessageBar.setOnClickListener(new View.OnClickListener() { |
| 318 | @Override |
| 319 | public void onClick(View v) { |
| 320 | onNewMessageBarClick(); |
| 321 | } |
| 322 | }); |
| 323 | |
mindyp | ff282d0 | 2012-09-17 10:33:02 -0700 | [diff] [blame] | 324 | instantiateProgressIndicators(rootView); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 325 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 326 | mWebView = (ConversationWebView) mConversationContainer.findViewById(R.id.webview); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 327 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 328 | mWebView.addJavascriptInterface(mJsBridge, "mail"); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 329 | // On JB or newer, we use the 'webkitAnimationStart' DOM event to signal load complete |
| 330 | // Below JB, try to speed up initial render by having the webview do supplemental draws to |
| 331 | // custom a software canvas. |
mindyp | b941fdb | 2012-09-11 08:28:23 -0700 | [diff] [blame] | 332 | // TODO(mindyp): |
| 333 | //PAGE READINESS SIGNAL FOR JELLYBEAN AND NEWER |
| 334 | // Notify the app on 'webkitAnimationStart' of a simple dummy element with a simple no-op |
| 335 | // animation that immediately runs on page load. The app uses this as a signal that the |
| 336 | // content is loaded and ready to draw, since WebView delays firing this event until the |
| 337 | // layers are composited and everything is ready to draw. |
| 338 | // This signal does not seem to be reliable, so just use the old method for now. |
mindyp | 32d911f | 2012-09-24 15:14:22 -0700 | [diff] [blame] | 339 | mEnableContentReadySignal = Utils.isRunningJellybeanOrLater(); |
mindyp | afc9b36 | 2012-09-25 09:20:47 -0700 | [diff] [blame] | 340 | mWebView.setUseSoftwareLayer(!mEnableContentReadySignal); |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 341 | mWebView.onUserVisibilityChanged(isUserVisible()); |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 342 | mWebView.setWebViewClient(mWebViewClient); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 343 | mWebView.setWebChromeClient(new WebChromeClient() { |
| 344 | @Override |
| 345 | public boolean onConsoleMessage(ConsoleMessage consoleMessage) { |
| 346 | LogUtils.i(LOG_TAG, "JS: %s (%s:%d)", consoleMessage.message(), |
| 347 | consoleMessage.sourceId(), consoleMessage.lineNumber()); |
| 348 | return true; |
| 349 | } |
| 350 | }); |
| 351 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 352 | final WebSettings settings = mWebView.getSettings(); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 353 | |
Mark Wei | 56d8385 | 2012-09-19 14:28:50 -0700 | [diff] [blame] | 354 | mScrollIndicators = (ScrollIndicatorsView) rootView.findViewById(R.id.scroll_indicators); |
| 355 | mScrollIndicators.setSourceView(mWebView); |
| 356 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 357 | settings.setJavaScriptEnabled(true); |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 358 | |
Andy Huang | c319b55 | 2012-04-25 19:53:50 -0700 | [diff] [blame] | 359 | final float fontScale = getResources().getConfiguration().fontScale; |
Andy Huang | ba28373 | 2012-06-25 19:14:10 -0700 | [diff] [blame] | 360 | final int desiredFontSizePx = getResources() |
| 361 | .getInteger(R.integer.conversation_desired_font_size_px); |
| 362 | final int unstyledFontSizePx = getResources() |
| 363 | .getInteger(R.integer.conversation_unstyled_font_size_px); |
Andy Huang | c319b55 | 2012-04-25 19:53:50 -0700 | [diff] [blame] | 364 | |
Andy Huang | ba28373 | 2012-06-25 19:14:10 -0700 | [diff] [blame] | 365 | int textZoom = settings.getTextZoom(); |
| 366 | // apply a correction to the default body text style to get regular text to the size we want |
| 367 | textZoom = textZoom * desiredFontSizePx / unstyledFontSizePx; |
| 368 | // then apply any system font scaling |
Andy Huang | c319b55 | 2012-04-25 19:53:50 -0700 | [diff] [blame] | 369 | textZoom = (int) (textZoom * fontScale); |
| 370 | settings.setTextZoom(textZoom); |
| 371 | |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 372 | mViewsCreated = true; |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 373 | mWebViewLoadedData = false; |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 374 | |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 375 | return rootView; |
| 376 | } |
| 377 | |
| 378 | @Override |
| 379 | public void onDestroyView() { |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 380 | super.onDestroyView(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 381 | mConversationContainer.setOverlayAdapter(null); |
| 382 | mAdapter = null; |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 383 | resetLoadWaiting(); // be sure to unregister any active load observer |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 384 | mViewsCreated = false; |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 385 | } |
| 386 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 387 | @Override |
| 388 | public void onSaveInstanceState(Bundle outState) { |
| 389 | super.onSaveInstanceState(outState); |
| 390 | |
| 391 | outState.putFloat(BUNDLE_KEY_WEBVIEW_Y_PERCENT, calculateScrollYPercent()); |
| 392 | } |
| 393 | |
| 394 | private float calculateScrollYPercent() { |
| 395 | float p; |
| 396 | int scrollY = mWebView.getScrollY(); |
| 397 | int viewH = mWebView.getHeight(); |
| 398 | int webH = (int) (mWebView.getContentHeight() * mWebView.getScale()); |
| 399 | |
| 400 | if (webH == 0 || webH <= viewH) { |
| 401 | p = 0; |
| 402 | } else if (scrollY + viewH >= webH) { |
| 403 | // The very bottom is a special case, it acts as a stronger anchor than the scroll top |
| 404 | // at that point. |
| 405 | p = 1.0f; |
| 406 | } else { |
| 407 | p = (float) scrollY / webH; |
| 408 | } |
| 409 | return p; |
| 410 | } |
| 411 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 412 | private void resetLoadWaiting() { |
| 413 | if (mLoadWaitReason == LOAD_WAIT_FOR_INITIAL_CONVERSATION) { |
| 414 | getListController().unregisterConversationLoadedObserver(mLoadedObserver); |
| 415 | } |
| 416 | mLoadWaitReason = LOAD_NOW; |
| 417 | } |
| 418 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 419 | @Override |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 420 | protected void markUnread() { |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 421 | // Ignore unsafe calls made after a fragment is detached from an activity |
| 422 | final ControllableActivity activity = (ControllableActivity) getActivity(); |
| 423 | if (activity == null) { |
| 424 | LogUtils.w(LOG_TAG, "ignoring markUnread for conv=%s", mConversation.id); |
| 425 | return; |
| 426 | } |
| 427 | |
Andy Huang | 28e31e2 | 2012-07-26 16:33:15 -0700 | [diff] [blame] | 428 | if (mViewState == null) { |
| 429 | LogUtils.i(LOG_TAG, "ignoring markUnread for conv with no view state (%d)", |
| 430 | mConversation.id); |
| 431 | return; |
| 432 | } |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 433 | activity.getConversationUpdater().markConversationMessagesUnread(mConversation, |
Vikram Aggarwal | 4a878b6 | 2012-07-31 15:09:25 -0700 | [diff] [blame] | 434 | mViewState.getUnreadMessageUris(), mViewState.getConversationInfo()); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 435 | } |
| 436 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 437 | @Override |
| 438 | public void onUserVisibleHintChanged() { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 439 | final boolean userVisible = isUserVisible(); |
| 440 | |
| 441 | if (!userVisible) { |
mindyp | 32d911f | 2012-09-24 15:14:22 -0700 | [diff] [blame] | 442 | dismissLoadingStatus(); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 443 | } else if (mViewsCreated) { |
| 444 | if (getMessageCursor() != null) { |
| 445 | LogUtils.d(LOG_TAG, "Fragment is now user-visible, onConversationSeen: %s", this); |
| 446 | onConversationSeen(); |
| 447 | } else if (isLoadWaiting()) { |
| 448 | LogUtils.d(LOG_TAG, "Fragment is now user-visible, showing conversation: %s", this); |
| 449 | handleDelayedConversationLoad(); |
| 450 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 451 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 452 | |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 453 | if (mWebView != null) { |
| 454 | mWebView.onUserVisibilityChanged(userVisible); |
| 455 | } |
Andy Huang | f8cf546 | 2012-10-17 18:29:14 -0700 | [diff] [blame] | 456 | } |
| 457 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 458 | /** |
| 459 | * Will either call initLoader now to begin loading, or set {@link #mLoadWaitReason} and do |
| 460 | * nothing (in which case you should later call {@link #handleDelayedConversationLoad()}). |
| 461 | */ |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 462 | private void showConversation() { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 463 | final int reason; |
| 464 | |
| 465 | if (isUserVisible()) { |
| 466 | LogUtils.i(LOG_TAG, |
| 467 | "SHOWCONV: CVF is user-visible, immediately loading conversation (%s)", this); |
| 468 | reason = LOAD_NOW; |
| 469 | } else { |
| 470 | final boolean disableOffscreenLoading = DISABLE_OFFSCREEN_LOADING |
| 471 | || (mConversation.isRemote |
| 472 | || mConversation.getNumMessages() > mMaxAutoLoadMessages); |
| 473 | |
| 474 | // When not visible, we should not immediately load if either this conversation is |
| 475 | // too heavyweight, or if the main/initial conversation is busy loading. |
| 476 | if (disableOffscreenLoading) { |
| 477 | reason = LOAD_WAIT_UNTIL_VISIBLE; |
| 478 | LogUtils.i(LOG_TAG, "SHOWCONV: CVF waiting until visible to load (%s)", this); |
| 479 | } else if (getListController().isInitialConversationLoading()) { |
| 480 | reason = LOAD_WAIT_FOR_INITIAL_CONVERSATION; |
| 481 | LogUtils.i(LOG_TAG, "SHOWCONV: CVF waiting for initial to finish (%s)", this); |
| 482 | getListController().registerConversationLoadedObserver(mLoadedObserver); |
| 483 | } else { |
| 484 | LogUtils.i(LOG_TAG, |
| 485 | "SHOWCONV: CVF is not visible, but no reason to wait. loading now. (%s)", |
| 486 | this); |
| 487 | reason = LOAD_NOW; |
| 488 | } |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 489 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 490 | |
| 491 | mLoadWaitReason = reason; |
| 492 | if (mLoadWaitReason == LOAD_NOW) { |
| 493 | startConversationLoad(); |
| 494 | } |
| 495 | } |
| 496 | |
| 497 | private void handleDelayedConversationLoad() { |
| 498 | resetLoadWaiting(); |
| 499 | startConversationLoad(); |
| 500 | } |
| 501 | |
| 502 | private void startConversationLoad() { |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 503 | mWebView.setVisibility(View.VISIBLE); |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 504 | getLoaderManager().initLoader(MESSAGE_LOADER, Bundle.EMPTY, getMessageLoaderCallbacks()); |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 505 | if (isUserVisible()) { |
Andy Huang | 5460ce4 | 2012-08-16 19:38:27 -0700 | [diff] [blame] | 506 | final SubjectDisplayChanger sdc = mActivity.getSubjectDisplayChanger(); |
| 507 | if (sdc != null) { |
| 508 | sdc.setSubject(mConversation.subject); |
| 509 | } |
| 510 | } |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 511 | // TODO(mindyp): don't show loading status for a previously rendered |
| 512 | // conversation. Ielieve this is better done by making sure don't show loading status |
| 513 | // until XX ms have passed without loading completed. |
| 514 | showLoadingStatus(); |
Mindy Pereira | 8e91572 | 2012-02-16 14:42:56 -0800 | [diff] [blame] | 515 | } |
| 516 | |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 517 | private void revealConversation() { |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 518 | dismissLoadingStatus(mOnProgressDismiss); |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 519 | } |
| 520 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 521 | private boolean isLoadWaiting() { |
| 522 | return mLoadWaitReason != LOAD_NOW; |
| 523 | } |
| 524 | |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 525 | private void renderConversation(MessageCursor messageCursor) { |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 526 | final String convHtml = renderMessageBodies(messageCursor, mEnableContentReadySignal); |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 527 | |
| 528 | if (DEBUG_DUMP_CONVERSATION_HTML) { |
| 529 | java.io.FileWriter fw = null; |
| 530 | try { |
| 531 | fw = new java.io.FileWriter("/sdcard/conv" + mConversation.id |
| 532 | + ".html"); |
| 533 | fw.write(convHtml); |
| 534 | } catch (java.io.IOException e) { |
| 535 | e.printStackTrace(); |
| 536 | } finally { |
| 537 | if (fw != null) { |
| 538 | try { |
| 539 | fw.close(); |
| 540 | } catch (java.io.IOException e) { |
| 541 | e.printStackTrace(); |
| 542 | } |
| 543 | } |
| 544 | } |
| 545 | } |
| 546 | |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 547 | // save off existing scroll position before re-rendering |
| 548 | if (mWebViewLoadedData) { |
| 549 | mWebViewYPercent = calculateScrollYPercent(); |
| 550 | } |
| 551 | |
Andy Huang | bd544e3 | 2012-05-29 15:56:51 -0700 | [diff] [blame] | 552 | mWebView.loadDataWithBaseURL(mBaseUri, convHtml, "text/html", "utf-8", null); |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 553 | mWebViewLoadedData = true; |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 554 | mWebViewLoadStartMs = SystemClock.uptimeMillis(); |
Andy Huang | 5106713 | 2012-03-12 20:08:19 -0700 | [diff] [blame] | 555 | } |
| 556 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 557 | /** |
| 558 | * Populate the adapter with overlay views (message headers, super-collapsed blocks, a |
| 559 | * conversation header), and return an HTML document with spacer divs inserted for all overlays. |
| 560 | * |
| 561 | */ |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 562 | private String renderMessageBodies(MessageCursor messageCursor, |
| 563 | boolean enableContentReadySignal) { |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 564 | int pos = -1; |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 565 | |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 566 | LogUtils.d(LOG_TAG, "IN renderMessageBodies, fragment=%s", this); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 567 | boolean allowNetworkImages = false; |
| 568 | |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 569 | // 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] | 570 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 571 | // Walk through the cursor and build up an overlay adapter as you go. |
| 572 | // Each overlay has an entry in the adapter for easy scroll handling in the container. |
| 573 | // Items are not necessarily 1:1 in cursor and adapter because of super-collapsed blocks. |
| 574 | // When adding adapter items, also add their heights to help the container later determine |
| 575 | // overlay dimensions. |
| 576 | |
Andy Huang | db620fe | 2012-08-24 15:45:28 -0700 | [diff] [blame] | 577 | // When re-rendering, prevent ConversationContainer from laying out overlays until after |
| 578 | // the new spacers are positioned by WebView. |
| 579 | mConversationContainer.invalidateSpacerGeometry(); |
| 580 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 581 | mAdapter.clear(); |
| 582 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 583 | // re-evaluate the message parts of the view state, since the messages may have changed |
| 584 | // since the previous render |
| 585 | final ConversationViewState prevState = mViewState; |
| 586 | mViewState = new ConversationViewState(prevState); |
| 587 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 588 | // 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] | 589 | // a pixel is an mdpi pixel, unless you set device-dpi. |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 590 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 591 | // add a single conversation header item |
| 592 | final int convHeaderPos = mAdapter.addConversationHeader(mConversation); |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 593 | final int convHeaderPx = measureOverlayHeight(convHeaderPos); |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 594 | |
Andy Huang | 256b35c | 2012-08-22 15:19:13 -0700 | [diff] [blame] | 595 | final int sideMarginPx = getResources().getDimensionPixelOffset( |
| 596 | R.dimen.conversation_view_margin_side) + getResources().getDimensionPixelOffset( |
| 597 | R.dimen.conversation_message_content_margin_side); |
| 598 | |
| 599 | mTemplates.startConversation(mWebView.screenPxToWebPx(sideMarginPx), |
| 600 | mWebView.screenPxToWebPx(convHeaderPx)); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 601 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 602 | int collapsedStart = -1; |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 603 | ConversationMessage prevCollapsedMsg = null; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 604 | boolean prevSafeForImages = false; |
| 605 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 606 | while (messageCursor.moveToPosition(++pos)) { |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 607 | final ConversationMessage msg = messageCursor.getMessage(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 608 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 609 | // TODO: save/restore 'show pics' state |
| 610 | final boolean safeForImages = msg.alwaysShowImages /* || savedStateSaysSafe */; |
| 611 | allowNetworkImages |= safeForImages; |
Andy Huang | 2405528 | 2012-03-27 17:37:06 -0700 | [diff] [blame] | 612 | |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 613 | final Integer savedExpanded = prevState.getExpansionState(msg); |
| 614 | final int expandedState; |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 615 | if (savedExpanded != null) { |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 616 | if (ExpansionState.isSuperCollapsed(savedExpanded) && messageCursor.isLast()) { |
| 617 | // override saved state when this is now the new last message |
| 618 | // this happens to the second-to-last message when you discard a draft |
| 619 | expandedState = ExpansionState.EXPANDED; |
| 620 | } else { |
| 621 | expandedState = savedExpanded; |
| 622 | } |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 623 | } else { |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 624 | // 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] | 625 | expandedState = (!msg.read || msg.starred || messageCursor.isLast()) ? |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 626 | ExpansionState.EXPANDED : ExpansionState.SUPER_COLLAPSED; |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 627 | } |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 628 | mViewState.setExpansionState(msg, expandedState); |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 629 | |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 630 | // save off "read" state from the cursor |
| 631 | // 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] | 632 | // however, if a previous state indicated this message was unread, trust that instead |
| 633 | // so "mark unread" marks all originally unread messages |
| 634 | mViewState.setReadState(msg, msg.read && !prevState.isUnread(msg)); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 635 | |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 636 | // We only want to consider this for inclusion in the super collapsed block if |
| 637 | // 1) The we don't have previous state about this message (The first time that the |
| 638 | // user opens a conversation) |
| 639 | // 2) The previously saved state for this message indicates that this message is |
| 640 | // in the super collapsed block. |
| 641 | if (ExpansionState.isSuperCollapsed(expandedState)) { |
| 642 | // contribute to a super-collapsed block that will be emitted just before the |
| 643 | // next expanded header |
| 644 | if (collapsedStart < 0) { |
| 645 | collapsedStart = pos; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 646 | } |
Andy Huang | cd5c5ee | 2012-08-12 19:03:51 -0700 | [diff] [blame] | 647 | prevCollapsedMsg = msg; |
| 648 | prevSafeForImages = safeForImages; |
| 649 | continue; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 650 | } |
Andy Huang | 2405528 | 2012-03-27 17:37:06 -0700 | [diff] [blame] | 651 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 652 | // resolve any deferred decisions on previous collapsed items |
| 653 | if (collapsedStart >= 0) { |
| 654 | if (pos - collapsedStart == 1) { |
| 655 | // special-case for a single collapsed message: no need to super-collapse it |
| 656 | renderMessage(prevCollapsedMsg, false /* expanded */, |
| 657 | prevSafeForImages); |
| 658 | } else { |
| 659 | renderSuperCollapsedBlock(collapsedStart, pos - 1); |
| 660 | } |
| 661 | prevCollapsedMsg = null; |
| 662 | collapsedStart = -1; |
| 663 | } |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 664 | |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 665 | renderMessage(msg, ExpansionState.isExpanded(expandedState), safeForImages); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 666 | } |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 667 | |
| 668 | mWebView.getSettings().setBlockNetworkImage(!allowNetworkImages); |
| 669 | |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 670 | // If the conversation has specified a base uri, use it here, use mBaseUri |
| 671 | final String conversationBaseUri = mConversation.conversationBaseUri != null ? |
| 672 | mConversation.conversationBaseUri.toString() : mBaseUri; |
| 673 | return mTemplates.endConversation(mBaseUri, conversationBaseUri, 320, |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 674 | mWebView.getViewportWidth(), enableContentReadySignal, isOverviewMode(mAccount)); |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 675 | } |
Mindy Pereira | 674afa4 | 2012-02-17 14:05:24 -0800 | [diff] [blame] | 676 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 677 | private void renderSuperCollapsedBlock(int start, int end) { |
| 678 | final int blockPos = mAdapter.addSuperCollapsedBlock(start, end); |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 679 | final int blockPx = measureOverlayHeight(blockPos); |
| 680 | mTemplates.appendSuperCollapsedHtml(start, mWebView.screenPxToWebPx(blockPx)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 681 | } |
| 682 | |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 683 | private void renderMessage(ConversationMessage msg, boolean expanded, |
| 684 | boolean safeForImages) { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 685 | final int headerPos = mAdapter.addMessageHeader(msg, expanded); |
| 686 | final MessageHeaderItem headerItem = (MessageHeaderItem) mAdapter.getItem(headerPos); |
| 687 | |
| 688 | final int footerPos = mAdapter.addMessageFooter(headerItem); |
| 689 | |
| 690 | // Measure item header and footer heights to allocate spacers in HTML |
| 691 | // But since the views themselves don't exist yet, render each item temporarily into |
| 692 | // a host view for measurement. |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 693 | final int headerPx = measureOverlayHeight(headerPos); |
| 694 | final int footerPx = measureOverlayHeight(footerPos); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 695 | |
Andy Huang | 256b35c | 2012-08-22 15:19:13 -0700 | [diff] [blame] | 696 | mTemplates.appendMessageHtml(msg, expanded, safeForImages, |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 697 | mWebView.screenPxToWebPx(headerPx), mWebView.screenPxToWebPx(footerPx)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 698 | } |
| 699 | |
| 700 | private String renderCollapsedHeaders(MessageCursor cursor, |
| 701 | SuperCollapsedBlockItem blockToReplace) { |
| 702 | final List<ConversationOverlayItem> replacements = Lists.newArrayList(); |
| 703 | |
| 704 | mTemplates.reset(); |
| 705 | |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 706 | // In devices with non-integral density multiplier, screen pixels translate to non-integral |
| 707 | // web pixels. Keep track of the error that occurs when we cast all heights to int |
| 708 | float error = 0f; |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 709 | for (int i = blockToReplace.getStart(), end = blockToReplace.getEnd(); i <= end; i++) { |
| 710 | cursor.moveToPosition(i); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 711 | final ConversationMessage msg = cursor.getMessage(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 712 | final MessageHeaderItem header = mAdapter.newMessageHeaderItem(msg, |
| 713 | false /* expanded */); |
| 714 | final MessageFooterItem footer = mAdapter.newMessageFooterItem(header); |
| 715 | |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 716 | final int headerPx = measureOverlayHeight(header); |
| 717 | final int footerPx = measureOverlayHeight(footer); |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 718 | error += mWebView.screenPxToWebPxError(headerPx) |
| 719 | + mWebView.screenPxToWebPxError(footerPx); |
| 720 | |
| 721 | // When the error becomes greater than 1 pixel, make the next header 1 pixel taller |
| 722 | int correction = 0; |
| 723 | if (error >= 1) { |
| 724 | correction = 1; |
| 725 | error -= 1; |
| 726 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 727 | |
Andy Huang | 256b35c | 2012-08-22 15:19:13 -0700 | [diff] [blame] | 728 | mTemplates.appendMessageHtml(msg, false /* expanded */, msg.alwaysShowImages, |
Mark Wei | 2b24e99 | 2012-09-10 16:40:07 -0700 | [diff] [blame] | 729 | mWebView.screenPxToWebPx(headerPx) + correction, |
| 730 | mWebView.screenPxToWebPx(footerPx)); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 731 | replacements.add(header); |
| 732 | replacements.add(footer); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 733 | |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 734 | mViewState.setExpansionState(msg, ExpansionState.COLLAPSED); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 735 | } |
| 736 | |
| 737 | mAdapter.replaceSuperCollapsedBlock(blockToReplace, replacements); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 738 | mAdapter.notifyDataSetChanged(); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 739 | |
| 740 | return mTemplates.emit(); |
| 741 | } |
| 742 | |
| 743 | private int measureOverlayHeight(int position) { |
| 744 | return measureOverlayHeight(mAdapter.getItem(position)); |
| 745 | } |
| 746 | |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 747 | /** |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 748 | * 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] | 749 | * 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] | 750 | * a previous adapter view from {@link ConversationContainer}'s scrap views if one was generated |
| 751 | * earlier. |
| 752 | * <p> |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 753 | * After measuring the height, this method also saves the height in the |
| 754 | * {@link ConversationOverlayItem} for later use in overlay positioning. |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 755 | * |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 756 | * @param convItem adapter item with data to render and measure |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 757 | * @return height of the rendered view in screen px |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 758 | */ |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 759 | private int measureOverlayHeight(ConversationOverlayItem convItem) { |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 760 | final int type = convItem.getType(); |
| 761 | |
| 762 | final View convertView = mConversationContainer.getScrapView(type); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 763 | final View hostView = mAdapter.getView(convItem, convertView, mConversationContainer, |
| 764 | true /* measureOnly */); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 765 | if (convertView == null) { |
| 766 | mConversationContainer.addScrapView(type, hostView); |
| 767 | } |
| 768 | |
Andy Huang | 9875bb4 | 2012-04-04 20:36:21 -0700 | [diff] [blame] | 769 | final int heightPx = mConversationContainer.measureOverlay(hostView); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 770 | convItem.setHeight(heightPx); |
Andy Huang | 9875bb4 | 2012-04-04 20:36:21 -0700 | [diff] [blame] | 771 | convItem.markMeasurementValid(); |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 772 | |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 773 | return heightPx; |
Andy Huang | 7bdc375 | 2012-03-25 17:18:19 -0700 | [diff] [blame] | 774 | } |
| 775 | |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 776 | @Override |
| 777 | public void onConversationViewHeaderHeightChange(int newHeight) { |
Mark Wei | ab2d998 | 2012-09-25 13:06:17 -0700 | [diff] [blame] | 778 | final int h = mWebView.screenPxToWebPx(newHeight); |
| 779 | |
| 780 | mWebView.loadUrl(String.format("javascript:setConversationHeaderSpacerHeight(%s);", h)); |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 781 | } |
| 782 | |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 783 | // END conversation header callbacks |
| 784 | |
| 785 | // START message header callbacks |
| 786 | @Override |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 787 | public void setMessageSpacerHeight(MessageHeaderItem item, int newSpacerHeightPx) { |
| 788 | mConversationContainer.invalidateSpacerGeometry(); |
| 789 | |
| 790 | // update message HTML spacer height |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 791 | final int h = mWebView.screenPxToWebPx(newSpacerHeightPx); |
| 792 | LogUtils.i(LAYOUT_TAG, "setting HTML spacer h=%dwebPx (%dscreenPx)", h, |
| 793 | newSpacerHeightPx); |
Vikram Aggarwal | 5349ce1 | 2012-09-24 14:12:40 -0700 | [diff] [blame] | 794 | mWebView.loadUrl(String.format("javascript:setMessageHeaderSpacerHeight('%s', %s);", |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 795 | mTemplates.getMessageDomId(item.getMessage()), h)); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 796 | } |
| 797 | |
| 798 | @Override |
Andy Huang | c754357 | 2012-04-03 15:34:29 -0700 | [diff] [blame] | 799 | public void setMessageExpanded(MessageHeaderItem item, int newSpacerHeightPx) { |
| 800 | mConversationContainer.invalidateSpacerGeometry(); |
| 801 | |
| 802 | // show/hide the HTML message body and update the spacer height |
Andy Huang | 2301470 | 2012-07-09 12:50:36 -0700 | [diff] [blame] | 803 | final int h = mWebView.screenPxToWebPx(newSpacerHeightPx); |
| 804 | LogUtils.i(LAYOUT_TAG, "setting HTML spacer expanded=%s h=%dwebPx (%dscreenPx)", |
| 805 | item.isExpanded(), h, newSpacerHeightPx); |
Vikram Aggarwal | 5349ce1 | 2012-09-24 14:12:40 -0700 | [diff] [blame] | 806 | mWebView.loadUrl(String.format("javascript:setMessageBodyVisible('%s', %s, %s);", |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 807 | mTemplates.getMessageDomId(item.getMessage()), item.isExpanded(), h)); |
Andy Huang | 839ada2 | 2012-07-20 15:48:40 -0700 | [diff] [blame] | 808 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 809 | mViewState.setExpansionState(item.getMessage(), |
Paul Westbrook | 08098ec | 2012-08-12 15:30:28 -0700 | [diff] [blame] | 810 | item.isExpanded() ? ExpansionState.EXPANDED : ExpansionState.COLLAPSED); |
Andy Huang | 3233bff | 2012-03-20 19:38:45 -0700 | [diff] [blame] | 811 | } |
| 812 | |
| 813 | @Override |
| 814 | public void showExternalResources(Message msg) { |
| 815 | mWebView.getSettings().setBlockNetworkImage(false); |
| 816 | mWebView.loadUrl("javascript:unblockImages('" + mTemplates.getMessageDomId(msg) + "');"); |
| 817 | } |
| 818 | // END message header callbacks |
Andy Huang | 5ff6374 | 2012-03-16 20:30:23 -0700 | [diff] [blame] | 819 | |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 820 | @Override |
| 821 | public void onSuperCollapsedClick(SuperCollapsedBlockItem item) { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 822 | MessageCursor cursor = getMessageCursor(); |
| 823 | if (cursor == null || !mViewsCreated) { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 824 | return; |
| 825 | } |
| 826 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 827 | mTempBodiesHtml = renderCollapsedHeaders(cursor, item); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 828 | mWebView.loadUrl("javascript:replaceSuperCollapsedBlock(" + item.getStart() + ")"); |
| 829 | } |
| 830 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 831 | private void showNewMessageNotification(NewMessagesInfo info) { |
| 832 | final TextView descriptionView = (TextView) mNewMessageBar.findViewById( |
| 833 | R.id.new_message_description); |
| 834 | descriptionView.setText(info.getNotificationText()); |
| 835 | mNewMessageBar.setVisibility(View.VISIBLE); |
| 836 | } |
| 837 | |
| 838 | private void onNewMessageBarClick() { |
| 839 | mNewMessageBar.setVisibility(View.GONE); |
| 840 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 841 | renderConversation(getMessageCursor()); // mCursor is already up-to-date |
| 842 | // per onLoadFinished() |
Andy Huang | 5fbda02 | 2012-02-28 18:22:03 -0800 | [diff] [blame] | 843 | } |
| 844 | |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 845 | private static OverlayPosition[] parsePositions(final String[] topArray, |
| 846 | final String[] bottomArray) { |
| 847 | final int len = topArray.length; |
| 848 | final OverlayPosition[] positions = new OverlayPosition[len]; |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 849 | for (int i = 0; i < len; i++) { |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 850 | positions[i] = new OverlayPosition( |
| 851 | Integer.parseInt(topArray[i]), Integer.parseInt(bottomArray[i])); |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 852 | } |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 853 | return positions; |
Andy Huang | b5078b2 | 2012-03-05 19:52:29 -0800 | [diff] [blame] | 854 | } |
| 855 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 856 | @Override |
| 857 | public String toString() { |
| 858 | // log extra info at DEBUG level or finer |
| 859 | final String s = super.toString(); |
| 860 | if (!LogUtils.isLoggable(LOG_TAG, LogUtils.DEBUG) || mConversation == null) { |
| 861 | return s; |
| 862 | } |
| 863 | return "(" + s + " subj=" + mConversation.subject + ")"; |
| 864 | } |
| 865 | |
Andy Huang | 1617481 | 2012-08-16 16:40:35 -0700 | [diff] [blame] | 866 | private Address getAddress(String rawFrom) { |
| 867 | Address addr = mAddressCache.get(rawFrom); |
| 868 | if (addr == null) { |
| 869 | addr = Address.getEmailAddress(rawFrom); |
| 870 | mAddressCache.put(rawFrom, addr); |
| 871 | } |
| 872 | return addr; |
| 873 | } |
| 874 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 875 | private void ensureContentSizeChangeListener() { |
| 876 | if (mWebViewSizeChangeListener == null) { |
| 877 | mWebViewSizeChangeListener = new ConversationWebView.ContentSizeChangeListener() { |
| 878 | @Override |
| 879 | public void onHeightChange(int h) { |
| 880 | // When WebKit says the DOM height has changed, re-measure |
| 881 | // bodies and re-position their headers. |
| 882 | // This is separate from the typical JavaScript DOM change |
| 883 | // listeners because cases like NARROW_COLUMNS text reflow do not trigger DOM |
| 884 | // events. |
| 885 | mWebView.loadUrl("javascript:measurePositions();"); |
| 886 | } |
| 887 | }; |
| 888 | } |
| 889 | mWebView.setContentSizeChangeListener(mWebViewSizeChangeListener); |
| 890 | } |
| 891 | |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 892 | private static boolean isOverviewMode(Account acct) { |
| 893 | return acct.settings.conversationViewMode == UIProvider.ConversationViewMode.OVERVIEW; |
| 894 | } |
| 895 | |
| 896 | private void setupOverviewMode() { |
| 897 | final boolean overviewMode = isOverviewMode(mAccount); |
| 898 | final WebSettings settings = mWebView.getSettings(); |
Andy Huang | 06def56 | 2012-10-14 00:19:11 -0700 | [diff] [blame] | 899 | settings.setUseWideViewPort(overviewMode); |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 900 | settings.setSupportZoom(overviewMode); |
| 901 | if (overviewMode) { |
| 902 | settings.setBuiltInZoomControls(true); |
| 903 | settings.setDisplayZoomControls(false); |
| 904 | } |
| 905 | } |
| 906 | |
Paul Westbrook | 542fec9 | 2012-09-18 14:47:51 -0700 | [diff] [blame] | 907 | private class ConversationWebViewClient extends AbstractConversationWebViewClient { |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 908 | @Override |
| 909 | public void onPageFinished(WebView view, String url) { |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 910 | // Ignore unsafe calls made after a fragment is detached from an activity. |
| 911 | // This method needs to, for example, get at the loader manager, which needs |
| 912 | // the fragment to be added. |
Andy Huang | de56e97 | 2012-07-26 18:23:08 -0700 | [diff] [blame] | 913 | final ControllableActivity activity = (ControllableActivity) getActivity(); |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 914 | if (!isAdded() || !mViewsCreated) { |
Andy Huang | de56e97 | 2012-07-26 18:23:08 -0700 | [diff] [blame] | 915 | LogUtils.i(LOG_TAG, "ignoring CVF.onPageFinished, url=%s fragment=%s", url, |
Andy Huang | b95da85 | 2012-07-18 14:16:58 -0700 | [diff] [blame] | 916 | ConversationViewFragment.this); |
| 917 | return; |
| 918 | } |
| 919 | |
Andy Huang | 30bcfe7 | 2012-10-18 18:09:03 -0700 | [diff] [blame] | 920 | LogUtils.i(LOG_TAG, "IN CVF.onPageFinished, url=%s fragment=%s wv=%s t=%sms", url, |
| 921 | ConversationViewFragment.this, view, |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 922 | (SystemClock.uptimeMillis() - mWebViewLoadStartMs)); |
Andy Huang | 632721e | 2012-04-11 16:57:26 -0700 | [diff] [blame] | 923 | |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 924 | ensureContentSizeChangeListener(); |
| 925 | |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 926 | if (!mEnableContentReadySignal) { |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 927 | revealConversation(); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 928 | } |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 929 | |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 930 | final Set<String> emailAddresses = Sets.newHashSet(); |
| 931 | for (Address addr : mAddressCache.values()) { |
| 932 | emailAddresses.add(addr.getAddress()); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 933 | } |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 934 | ContactLoaderCallbacks callbacks = getContactInfoSource(); |
| 935 | getContactInfoSource().setSenders(emailAddresses); |
| 936 | getLoaderManager().restartLoader(CONTACT_LOADER, Bundle.EMPTY, callbacks); |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 937 | } |
| 938 | |
Andy Huang | af5d4e0 | 2012-03-19 19:02:12 -0700 | [diff] [blame] | 939 | @Override |
| 940 | public boolean shouldOverrideUrlLoading(WebView view, String url) { |
Paul Westbrook | 542fec9 | 2012-09-18 14:47:51 -0700 | [diff] [blame] | 941 | return mViewsCreated && super.shouldOverrideUrlLoading(view, url); |
Andy Huang | af5d4e0 | 2012-03-19 19:02:12 -0700 | [diff] [blame] | 942 | } |
Andy Huang | 17a9cde | 2012-03-09 18:03:16 -0800 | [diff] [blame] | 943 | } |
| 944 | |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 945 | /** |
| 946 | * NOTE: all public methods must be listed in the proguard flags so that they can be accessed |
| 947 | * via reflection and not stripped. |
| 948 | * |
| 949 | */ |
| 950 | private class MailJsBridge { |
| 951 | |
| 952 | @SuppressWarnings("unused") |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 953 | @JavascriptInterface |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 954 | public void onWebContentGeometryChange(final String[] overlayTopStrs, |
| 955 | final String[] overlayBottomStrs) { |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 956 | getHandler().post(new FragmentRunnable("onWebContentGeometryChange") { |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 957 | |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 958 | @Override |
| 959 | public void go() { |
| 960 | try { |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 961 | if (!mViewsCreated) { |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 962 | LogUtils.d(LOG_TAG, "ignoring webContentGeometryChange because views" |
| 963 | + " are gone, %s", ConversationViewFragment.this); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 964 | return; |
| 965 | } |
Andy Huang | adbf3e8 | 2012-10-13 13:30:19 -0700 | [diff] [blame] | 966 | mConversationContainer.onGeometryChange( |
| 967 | parsePositions(overlayTopStrs, overlayBottomStrs)); |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 968 | if (mDiff != 0) { |
| 969 | // SCROLL! |
| 970 | int scale = (int) (mWebView.getScale() / mWebView.getInitialScale()); |
| 971 | if (scale > 1) { |
| 972 | mWebView.scrollBy(0, (mDiff * (scale - 1))); |
| 973 | } |
| 974 | mDiff = 0; |
| 975 | } |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 976 | } catch (Throwable t) { |
| 977 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onWebContentGeometryChange"); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 978 | } |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 979 | } |
| 980 | }); |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 981 | } |
| 982 | |
| 983 | @SuppressWarnings("unused") |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 984 | @JavascriptInterface |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 985 | public String getTempMessageBodies() { |
| 986 | try { |
| 987 | if (!mViewsCreated) { |
| 988 | return ""; |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 989 | } |
Andy Huang | 46dfba6 | 2012-04-19 01:47:32 -0700 | [diff] [blame] | 990 | |
| 991 | final String s = mTempBodiesHtml; |
| 992 | mTempBodiesHtml = null; |
| 993 | return s; |
| 994 | } catch (Throwable t) { |
| 995 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getTempMessageBodies"); |
| 996 | return ""; |
| 997 | } |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 998 | } |
| 999 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1000 | @SuppressWarnings("unused") |
| 1001 | @JavascriptInterface |
| 1002 | public String getMessageBody(String domId) { |
| 1003 | try { |
| 1004 | final MessageCursor cursor = getMessageCursor(); |
| 1005 | if (!mViewsCreated || cursor == null) { |
| 1006 | return ""; |
| 1007 | } |
| 1008 | |
| 1009 | int pos = -1; |
| 1010 | while (cursor.moveToPosition(++pos)) { |
| 1011 | final ConversationMessage msg = cursor.getMessage(); |
| 1012 | if (TextUtils.equals(domId, mTemplates.getMessageDomId(msg))) { |
| 1013 | return msg.getBodyAsHtml(); |
| 1014 | } |
| 1015 | } |
| 1016 | |
| 1017 | return ""; |
| 1018 | |
| 1019 | } catch (Throwable t) { |
| 1020 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getMessageBody"); |
| 1021 | return ""; |
| 1022 | } |
| 1023 | } |
| 1024 | |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1025 | @SuppressWarnings("unused") |
Mindy Pereira | 974c966 | 2012-09-14 10:02:08 -0700 | [diff] [blame] | 1026 | @JavascriptInterface |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1027 | public void onContentReady() { |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1028 | getHandler().post(new FragmentRunnable("onContentReady") { |
| 1029 | @Override |
| 1030 | public void go() { |
| 1031 | try { |
Andy Huang | 63b3c67 | 2012-10-05 19:27:28 -0700 | [diff] [blame] | 1032 | if (mWebViewLoadStartMs != 0) { |
| 1033 | LogUtils.i(LOG_TAG, "IN CVF.onContentReady, f=%s vis=%s t=%sms", |
| 1034 | ConversationViewFragment.this, |
| 1035 | isUserVisible(), |
| 1036 | (SystemClock.uptimeMillis() - mWebViewLoadStartMs)); |
| 1037 | } |
Andy Huang | 7d4746e | 2012-10-17 17:03:17 -0700 | [diff] [blame] | 1038 | revealConversation(); |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1039 | } catch (Throwable t) { |
| 1040 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.onContentReady"); |
| 1041 | // Still try to show the conversation. |
| 1042 | revealConversation(); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1043 | } |
Andy Huang | 9a8bc1e | 2012-10-23 19:48:25 -0700 | [diff] [blame] | 1044 | } |
| 1045 | }); |
mindyp | 3bcf180 | 2012-09-09 11:17:00 -0700 | [diff] [blame] | 1046 | } |
Andy Huang | e964eee | 2012-10-02 19:24:58 -0700 | [diff] [blame] | 1047 | |
| 1048 | @SuppressWarnings("unused") |
| 1049 | @JavascriptInterface |
| 1050 | public float getScrollYPercent() { |
| 1051 | try { |
| 1052 | return mWebViewYPercent; |
| 1053 | } catch (Throwable t) { |
| 1054 | LogUtils.e(LOG_TAG, t, "Error in MailJsBridge.getScrollYPercent"); |
| 1055 | return 0f; |
| 1056 | } |
| 1057 | } |
Andy Huang | f70fc40 | 2012-02-17 15:37:42 -0800 | [diff] [blame] | 1058 | } |
| 1059 | |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1060 | private class NewMessagesInfo { |
| 1061 | int count; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1062 | int countFromSelf; |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1063 | String senderAddress; |
| 1064 | |
| 1065 | /** |
| 1066 | * Return the display text for the new message notification overlay. It will be formatted |
| 1067 | * appropriately for a single new message vs. multiple new messages. |
| 1068 | * |
| 1069 | * @return display text |
| 1070 | */ |
| 1071 | public String getNotificationText() { |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 1072 | Resources res = getResources(); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1073 | if (count > 1) { |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 1074 | return res.getString(R.string.new_incoming_messages_many, count); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1075 | } else { |
Andy Huang | 1617481 | 2012-08-16 16:40:35 -0700 | [diff] [blame] | 1076 | final Address addr = getAddress(senderAddress); |
mindyp | ad0c30d | 2012-09-25 12:09:13 -0700 | [diff] [blame] | 1077 | return res.getString(R.string.new_incoming_messages_one, |
| 1078 | TextUtils.isEmpty(addr.getName()) ? addr.getAddress() : addr.getName()); |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1079 | } |
Andy Huang | 47aa9c9 | 2012-07-31 15:37:21 -0700 | [diff] [blame] | 1080 | } |
| 1081 | } |
| 1082 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1083 | @Override |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1084 | public void onMessageCursorLoadFinished(Loader<Cursor> loader, MessageCursor newCursor, |
| 1085 | MessageCursor oldCursor) { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1086 | /* |
| 1087 | * what kind of changes affect the MessageCursor? 1. new message(s) 2. |
| 1088 | * read/unread state change 3. deleted message, either regular or draft |
| 1089 | * 4. updated message, either from self or from others, updated in |
| 1090 | * content or state or sender 5. star/unstar of message (technically |
| 1091 | * similar to #1) 6. other label change Use MessageCursor.hashCode() to |
| 1092 | * sort out interesting vs. no-op cursor updates. |
| 1093 | */ |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1094 | |
Andy Huang | 233d435 | 2012-10-18 14:00:24 -0700 | [diff] [blame] | 1095 | if (oldCursor != null && !oldCursor.isClosed()) { |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1096 | final NewMessagesInfo info = getNewIncomingMessagesInfo(newCursor); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1097 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1098 | if (info.count > 0) { |
| 1099 | // don't immediately render new incoming messages from other |
| 1100 | // senders |
| 1101 | // (to avoid a new message from losing the user's focus) |
| 1102 | LogUtils.i(LOG_TAG, "CONV RENDER: conversation updated" |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1103 | + ", holding cursor for new incoming message (%s)", this); |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1104 | showNewMessageNotification(info); |
| 1105 | return; |
| 1106 | } |
| 1107 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1108 | final int oldState = oldCursor.getStateHashCode(); |
| 1109 | final boolean changed = newCursor.getStateHashCode() != oldState; |
Andy Huang | 233d435 | 2012-10-18 14:00:24 -0700 | [diff] [blame] | 1110 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1111 | if (!changed) { |
| 1112 | final boolean processedInPlace = processInPlaceUpdates(newCursor, oldCursor); |
| 1113 | if (processedInPlace) { |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1114 | 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] | 1115 | } else { |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1116 | LogUtils.i(LOG_TAG, "CONV RENDER: uninteresting update" |
Andy Huang | 9d3fd92 | 2012-09-26 22:23:58 -0700 | [diff] [blame] | 1117 | + ", ignoring this conversation update (%s)", this); |
Andy Huang | 1ee96b2 | 2012-08-24 20:19:53 -0700 | [diff] [blame] | 1118 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1119 | return; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1120 | } else if (info.countFromSelf == 1) { |
| 1121 | // Special-case the very common case of a new cursor that is the same as the old |
| 1122 | // one, except that there is a new message from yourself. This happens upon send. |
| 1123 | final boolean sameExceptNewLast = newCursor.getStateHashCode(1) == oldState; |
| 1124 | if (sameExceptNewLast) { |
| 1125 | LogUtils.i(LOG_TAG, "CONV RENDER: update is a single new message from self" |
| 1126 | + " (%s)", this); |
| 1127 | newCursor.moveToLast(); |
| 1128 | processNewOutgoingMessage(newCursor.getMessage()); |
| 1129 | return; |
| 1130 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1131 | } |
Andy Huang | 6766b6e | 2012-09-28 12:43:52 -0700 | [diff] [blame] | 1132 | // cursors are different, and not due to an incoming message. fall |
| 1133 | // through and render. |
| 1134 | LogUtils.i(LOG_TAG, "CONV RENDER: conversation updated" |
| 1135 | + ", but not due to incoming message. rendering. (%s)", this); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1136 | |
| 1137 | if (DEBUG_DUMP_CURSOR_CONTENTS) { |
| 1138 | LogUtils.i(LOG_TAG, "old cursor: %s", oldCursor.getDebugDump()); |
| 1139 | LogUtils.i(LOG_TAG, "new cursor: %s", newCursor.getDebugDump()); |
| 1140 | } |
Andy Huang | 6766b6e | 2012-09-28 12:43:52 -0700 | [diff] [blame] | 1141 | } else { |
| 1142 | LogUtils.i(LOG_TAG, "CONV RENDER: initial render. (%s)", this); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1143 | } |
| 1144 | |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 1145 | // if layout hasn't happened, delay render |
| 1146 | // This is needed in addition to the showConversation() delay to speed |
| 1147 | // up rotation and restoration. |
| 1148 | if (mConversationContainer.getWidth() == 0) { |
| 1149 | mNeedRender = true; |
| 1150 | mConversationContainer.addOnLayoutChangeListener(this); |
| 1151 | } else { |
| 1152 | renderConversation(newCursor); |
| 1153 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1154 | } |
| 1155 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1156 | private NewMessagesInfo getNewIncomingMessagesInfo(MessageCursor newCursor) { |
| 1157 | final NewMessagesInfo info = new NewMessagesInfo(); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1158 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1159 | int pos = -1; |
| 1160 | while (newCursor.moveToPosition(++pos)) { |
| 1161 | final Message m = newCursor.getMessage(); |
| 1162 | if (!mViewState.contains(m)) { |
| 1163 | LogUtils.i(LOG_TAG, "conversation diff: found new msg: %s", m.uri); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1164 | |
Scott Kennedy | 8960f0a | 2012-11-07 15:35:50 -0800 | [diff] [blame^] | 1165 | final Address from = getAddress(m.getFrom()); |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1166 | // distinguish ours from theirs |
| 1167 | // new messages from the account owner should not trigger a |
| 1168 | // notification |
| 1169 | if (mAccount.ownsFromAddress(from.getAddress())) { |
| 1170 | LogUtils.i(LOG_TAG, "found message from self: %s", m.uri); |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1171 | info.countFromSelf++; |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1172 | continue; |
| 1173 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1174 | |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1175 | info.count++; |
Scott Kennedy | 8960f0a | 2012-11-07 15:35:50 -0800 | [diff] [blame^] | 1176 | info.senderAddress = m.getFrom(); |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1177 | } |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1178 | } |
mindyp | f4fce12 | 2012-09-14 15:55:33 -0700 | [diff] [blame] | 1179 | return info; |
Andy Huang | b8331b4 | 2012-07-16 19:08:53 -0700 | [diff] [blame] | 1180 | } |
| 1181 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1182 | private boolean processInPlaceUpdates(MessageCursor newCursor, MessageCursor oldCursor) { |
| 1183 | final Set<String> idsOfChangedBodies = Sets.newHashSet(); |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1184 | final List<Integer> changedOverlayPositions = Lists.newArrayList(); |
| 1185 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1186 | boolean changed = false; |
| 1187 | |
| 1188 | int pos = 0; |
| 1189 | while (true) { |
| 1190 | if (!newCursor.moveToPosition(pos) || !oldCursor.moveToPosition(pos)) { |
| 1191 | break; |
| 1192 | } |
| 1193 | |
| 1194 | final ConversationMessage newMsg = newCursor.getMessage(); |
| 1195 | final ConversationMessage oldMsg = oldCursor.getMessage(); |
| 1196 | |
Scott Kennedy | 8960f0a | 2012-11-07 15:35:50 -0800 | [diff] [blame^] | 1197 | if (!TextUtils.equals(newMsg.getFrom(), oldMsg.getFrom()) || |
Andy Huang | 2a1e8e3 | 2012-10-23 18:54:57 -0700 | [diff] [blame] | 1198 | newMsg.isSending != oldMsg.isSending) { |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1199 | mAdapter.updateItemsForMessage(newMsg, changedOverlayPositions); |
Andy Huang | 2a1e8e3 | 2012-10-23 18:54:57 -0700 | [diff] [blame] | 1200 | LogUtils.i(LOG_TAG, "msg #%d (%d): detected from/sending change. isSending=%s", |
| 1201 | pos, newMsg.id, newMsg.isSending); |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1202 | } |
| 1203 | |
| 1204 | // update changed message bodies in-place |
| 1205 | if (!TextUtils.equals(newMsg.bodyHtml, oldMsg.bodyHtml) || |
| 1206 | !TextUtils.equals(newMsg.bodyText, oldMsg.bodyText)) { |
| 1207 | // maybe just set a flag to notify JS to re-request changed bodies |
| 1208 | idsOfChangedBodies.add('"' + mTemplates.getMessageDomId(newMsg) + '"'); |
| 1209 | LogUtils.i(LOG_TAG, "msg #%d (%d): detected body change", pos, newMsg.id); |
| 1210 | } |
| 1211 | |
| 1212 | pos++; |
| 1213 | } |
| 1214 | |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1215 | |
| 1216 | if (!changedOverlayPositions.isEmpty()) { |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1217 | // notify once after the entire adapter is updated |
Andy Huang | 6b3d0d9 | 2012-10-30 15:46:48 -0700 | [diff] [blame] | 1218 | mConversationContainer.onOverlayModelUpdate(changedOverlayPositions); |
| 1219 | changed = true; |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1220 | } |
| 1221 | |
Andy Huang | 014ea4c | 2012-09-25 14:50:54 -0700 | [diff] [blame] | 1222 | if (!idsOfChangedBodies.isEmpty()) { |
| 1223 | mWebView.loadUrl(String.format("javascript:replaceMessageBodies([%s]);", |
| 1224 | TextUtils.join(",", idsOfChangedBodies))); |
| 1225 | changed = true; |
| 1226 | } |
| 1227 | |
| 1228 | return changed; |
| 1229 | } |
| 1230 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1231 | private void processNewOutgoingMessage(ConversationMessage msg) { |
| 1232 | mTemplates.reset(); |
| 1233 | // this method will add some items to mAdapter, but we deliberately want to avoid notifying |
| 1234 | // adapter listeners (i.e. ConversationContainer) until onWebContentGeometryChange is next |
| 1235 | // called, to prevent N+1 headers rendering with N message bodies. |
| 1236 | renderMessage(msg, true /* expanded */, msg.alwaysShowImages); |
| 1237 | mTempBodiesHtml = mTemplates.emit(); |
| 1238 | |
| 1239 | mViewState.setExpansionState(msg, ExpansionState.EXPANDED); |
| 1240 | // FIXME: should the provider set this as initial state? |
| 1241 | mViewState.setReadState(msg, false /* read */); |
| 1242 | |
Andy Huang | 91d782a | 2012-10-25 12:37:29 -0700 | [diff] [blame] | 1243 | // From now until the updated spacer geometry is returned, the adapter items are mismatched |
| 1244 | // with the existing spacers. Do not let them layout. |
| 1245 | mConversationContainer.invalidateSpacerGeometry(); |
| 1246 | |
Andy Huang | 06c0362 | 2012-10-22 18:59:45 -0700 | [diff] [blame] | 1247 | mWebView.loadUrl("javascript:appendMessageHtml();"); |
| 1248 | } |
| 1249 | |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1250 | private class SetCookieTask extends AsyncTask<Void, Void, Void> { |
| 1251 | final String mUri; |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1252 | final Uri mAccountCookieQueryUri; |
| 1253 | final ContentResolver mResolver; |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1254 | |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1255 | SetCookieTask(Context context, Uri baseUri, Uri accountCookieQueryUri) { |
| 1256 | mUri = baseUri.toString(); |
| 1257 | mAccountCookieQueryUri = accountCookieQueryUri; |
| 1258 | mResolver = context.getContentResolver(); |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1259 | } |
| 1260 | |
| 1261 | @Override |
| 1262 | public Void doInBackground(Void... args) { |
Paul Westbrook | b8361c9 | 2012-09-27 10:57:14 -0700 | [diff] [blame] | 1263 | // First query for the coookie string from the UI provider |
| 1264 | final Cursor cookieCursor = mResolver.query(mAccountCookieQueryUri, |
| 1265 | UIProvider.ACCOUNT_COOKIE_PROJECTION, null, null, null); |
| 1266 | if (cookieCursor == null) { |
| 1267 | return null; |
| 1268 | } |
| 1269 | |
| 1270 | try { |
| 1271 | if (cookieCursor.moveToFirst()) { |
| 1272 | final String cookie = cookieCursor.getString( |
| 1273 | cookieCursor.getColumnIndex(UIProvider.AccountCookieColumns.COOKIE)); |
| 1274 | |
| 1275 | if (cookie != null) { |
| 1276 | final CookieSyncManager csm = |
| 1277 | CookieSyncManager.createInstance(getContext()); |
| 1278 | CookieManager.getInstance().setCookie(mUri, cookie); |
| 1279 | csm.sync(); |
| 1280 | } |
| 1281 | } |
| 1282 | |
| 1283 | } finally { |
| 1284 | cookieCursor.close(); |
| 1285 | } |
| 1286 | |
| 1287 | |
Paul Westbrook | cebcc64 | 2012-08-08 10:06:04 -0700 | [diff] [blame] | 1288 | return null; |
| 1289 | } |
| 1290 | } |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1291 | |
mindyp | 26d4d2d | 2012-09-18 17:30:32 -0700 | [diff] [blame] | 1292 | @Override |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1293 | public void onConversationUpdated(Conversation conv) { |
| 1294 | final ConversationViewHeader headerView = (ConversationViewHeader) mConversationContainer |
| 1295 | .findViewById(R.id.conversation_header); |
mindyp | b2b98ba | 2012-09-24 14:13:58 -0700 | [diff] [blame] | 1296 | mConversation = conv; |
mindyp | 9e0b236 | 2012-09-09 16:31:21 -0700 | [diff] [blame] | 1297 | if (headerView != null) { |
| 1298 | headerView.onConversationUpdated(conv); |
| 1299 | } |
mindyp | 36280f3 | 2012-09-09 16:11:23 -0700 | [diff] [blame] | 1300 | } |
Mark Wei | 4071c2f | 2012-09-26 14:38:38 -0700 | [diff] [blame] | 1301 | |
| 1302 | @Override |
| 1303 | public void onLayoutChange(View v, int left, int top, int right, |
| 1304 | int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) { |
| 1305 | boolean sizeChanged = mNeedRender |
| 1306 | && mConversationContainer.getWidth() != 0; |
| 1307 | if (sizeChanged) { |
| 1308 | mNeedRender = false; |
| 1309 | mConversationContainer.removeOnLayoutChangeListener(this); |
| 1310 | renderConversation(getMessageCursor()); |
| 1311 | } |
| 1312 | } |
mindyp | 1b3cc47 | 2012-09-27 11:32:59 -0700 | [diff] [blame] | 1313 | |
| 1314 | @Override |
| 1315 | public void setMessageDetailsExpanded(MessageHeaderItem i, boolean expanded, |
| 1316 | int heightBefore) { |
| 1317 | mDiff = (expanded ? 1 : -1) * Math.abs(i.getHeight() - heightBefore); |
| 1318 | } |
Mindy Pereira | 9b87568 | 2012-02-15 18:10:54 -0800 | [diff] [blame] | 1319 | } |