• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * Copyright (c) 2011, Google Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *     http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.mail.browse;
18 
19 import android.annotation.SuppressLint;
20 import android.app.FragmentManager;
21 import android.content.AsyncQueryHandler;
22 import android.content.Context;
23 import android.content.res.Resources;
24 import android.database.DataSetObserver;
25 import android.graphics.Bitmap;
26 import androidx.core.text.BidiFormatter;
27 import android.text.Html;
28 import android.text.Spannable;
29 import android.text.Spanned;
30 import android.text.TextUtils;
31 import android.text.method.LinkMovementMethod;
32 import android.text.style.URLSpan;
33 import android.util.AttributeSet;
34 import android.view.LayoutInflater;
35 import android.view.Menu;
36 import android.view.MenuItem;
37 import android.view.View;
38 import android.view.View.OnClickListener;
39 import android.view.ViewGroup;
40 import android.widget.PopupMenu;
41 import android.widget.PopupMenu.OnMenuItemClickListener;
42 import android.widget.QuickContactBadge;
43 import android.widget.TextView;
44 import android.widget.Toast;
45 
46 import com.android.emailcommon.mail.Address;
47 import com.android.mail.ContactInfo;
48 import com.android.mail.ContactInfoSource;
49 import com.android.mail.R;
50 import com.android.mail.analytics.Analytics;
51 import com.android.mail.browse.ConversationViewAdapter.MessageHeaderItem;
52 import com.android.mail.compose.ComposeActivity;
53 import com.android.mail.perf.Timer;
54 import com.android.mail.photomanager.LetterTileProvider;
55 import com.android.mail.print.PrintUtils;
56 import com.android.mail.providers.Account;
57 import com.android.mail.providers.Conversation;
58 import com.android.mail.providers.Message;
59 import com.android.mail.providers.Settings;
60 import com.android.mail.providers.UIProvider;
61 import com.android.mail.text.EmailAddressSpan;
62 import com.android.mail.ui.AbstractConversationViewFragment;
63 import com.android.mail.ui.ImageCanvas;
64 import com.android.mail.utils.BitmapUtil;
65 import com.android.mail.utils.LogTag;
66 import com.android.mail.utils.LogUtils;
67 import com.android.mail.utils.StyleUtils;
68 import com.android.mail.utils.Utils;
69 import com.android.mail.utils.VeiledAddressMatcher;
70 import com.google.common.annotations.VisibleForTesting;
71 
72 import java.io.IOException;
73 import java.io.StringReader;
74 import java.util.Map;
75 
76 public class MessageHeaderView extends SnapHeader implements OnClickListener,
77         OnMenuItemClickListener, ConversationContainer.DetachListener {
78 
79     /**
80      * Cap very long recipient lists during summary construction for efficiency.
81      */
82     private static final int SUMMARY_MAX_RECIPIENTS = 50;
83 
84     private static final int MAX_SNIPPET_LENGTH = 100;
85 
86     private static final int SHOW_IMAGE_PROMPT_ONCE = 1;
87     private static final int SHOW_IMAGE_PROMPT_ALWAYS = 2;
88 
89     private static final String HEADER_RENDER_TAG = "message header render";
90     private static final String LAYOUT_TAG = "message header layout";
91     private static final String MEASURE_TAG = "message header measure";
92 
93     private static final String LOG_TAG = LogTag.getLogTag();
94 
95     // This is a debug only feature
96     public static final boolean ENABLE_REPORT_RENDERING_PROBLEM = false;
97 
98     private MessageHeaderViewCallbacks mCallbacks;
99 
100     private View mBorderView;
101     private ViewGroup mUpperHeaderView;
102     private View mTitleContainer;
103     private View mSnapHeaderBottomBorder;
104     private TextView mSenderNameView;
105     private TextView mRecipientSummary;
106     private TextView mDateView;
107     private View mHideDetailsView;
108     private TextView mSnippetView;
109     private MessageHeaderContactBadge mPhotoView;
110     private ViewGroup mExtraContentView;
111     private ViewGroup mExpandedDetailsView;
112     private SpamWarningView mSpamWarningView;
113     private TextView mImagePromptView;
114     private MessageInviteView mInviteView;
115     private View mForwardButton;
116     private View mOverflowButton;
117     private View mDraftIcon;
118     private View mEditDraftButton;
119     private TextView mUpperDateView;
120     private View mReplyButton;
121     private View mReplyAllButton;
122     private View mAttachmentIcon;
123     private final EmailCopyContextMenu mEmailCopyMenu;
124 
125     // temporary fields to reference raw data between initial render and details
126     // expansion
127     private String[] mFrom;
128     private String[] mTo;
129     private String[] mCc;
130     private String[] mBcc;
131     private String[] mReplyTo;
132 
133     private boolean mIsDraft = false;
134 
135     private int mSendingState;
136 
137     private String mSnippet;
138 
139     private Address mSender;
140 
141     private ContactInfoSource mContactInfoSource;
142 
143     private boolean mPreMeasuring;
144 
145     private ConversationAccountController mAccountController;
146 
147     private Map<String, Address> mAddressCache;
148 
149     private boolean mShowImagePrompt;
150 
151     private PopupMenu mPopup;
152 
153     private MessageHeaderItem mMessageHeaderItem;
154     private ConversationMessage mMessage;
155 
156     private boolean mRecipientSummaryValid;
157     private boolean mExpandedDetailsValid;
158 
159     private final LayoutInflater mInflater;
160 
161     private AsyncQueryHandler mQueryHandler;
162 
163     private boolean mObservingContactInfo;
164 
165     /**
166      * What I call myself? "me" in English, and internationalized correctly.
167      */
168     private final String mMyName;
169 
170     private final DataSetObserver mContactInfoObserver = new DataSetObserver() {
171         @Override
172         public void onChanged() {
173             updateContactInfo();
174         }
175     };
176 
177     private boolean mExpandable = true;
178 
179     private VeiledAddressMatcher mVeiledMatcher;
180 
181     private boolean mIsViewOnlyMode = false;
182 
183     private LetterTileProvider mLetterTileProvider;
184     private final int mContactPhotoWidth;
185     private final int mContactPhotoHeight;
186     private final int mTitleContainerMarginEnd;
187 
188     /**
189      * The snappy header has special visibility rules (i.e. no details header,
190      * even though it has an expanded appearance)
191      */
192     private boolean mIsSnappy;
193 
194     private BidiFormatter mBidiFormatter;
195 
196 
197     public interface MessageHeaderViewCallbacks {
setMessageSpacerHeight(MessageHeaderItem item, int newSpacerHeight)198         void setMessageSpacerHeight(MessageHeaderItem item, int newSpacerHeight);
199 
setMessageExpanded(MessageHeaderItem item, int newSpacerHeight)200         void setMessageExpanded(MessageHeaderItem item, int newSpacerHeight);
201 
setMessageDetailsExpanded(MessageHeaderItem messageHeaderItem, boolean expanded, int previousMessageHeaderItemHeight)202         void setMessageDetailsExpanded(MessageHeaderItem messageHeaderItem, boolean expanded,
203                 int previousMessageHeaderItemHeight);
204 
showExternalResources(Message msg)205         void showExternalResources(Message msg);
206 
showExternalResources(String senderRawAddress)207         void showExternalResources(String senderRawAddress);
208 
supportsMessageTransforms()209         boolean supportsMessageTransforms();
210 
getMessageTransforms(Message msg)211         String getMessageTransforms(Message msg);
212 
getFragmentManager()213         FragmentManager getFragmentManager();
214 
215         /**
216          * @return <tt>true</tt> if this header is contained within a SecureConversationViewFragment
217          * and cannot assume the content is <strong>not</strong> malicious
218          */
isSecure()219         boolean isSecure();
220     }
221 
MessageHeaderView(Context context)222     public MessageHeaderView(Context context) {
223         this(context, null);
224     }
225 
MessageHeaderView(Context context, AttributeSet attrs)226     public MessageHeaderView(Context context, AttributeSet attrs) {
227         this(context, attrs, -1);
228     }
229 
MessageHeaderView(Context context, AttributeSet attrs, int defStyle)230     public MessageHeaderView(Context context, AttributeSet attrs, int defStyle) {
231         super(context, attrs, defStyle);
232 
233         mIsSnappy = false;
234         mEmailCopyMenu = new EmailCopyContextMenu(getContext());
235         mInflater = LayoutInflater.from(context);
236         mMyName = context.getString(R.string.me_object_pronoun);
237 
238         final Resources res = getResources();
239         mContactPhotoWidth = res.getDimensionPixelSize(R.dimen.contact_image_width);
240         mContactPhotoHeight = res.getDimensionPixelSize(R.dimen.contact_image_height);
241         mTitleContainerMarginEnd = res.getDimensionPixelSize(R.dimen.conversation_view_margin_side);
242     }
243 
244     @Override
onFinishInflate()245     protected void onFinishInflate() {
246         super.onFinishInflate();
247         mBorderView = findViewById(R.id.message_header_border);
248         mUpperHeaderView = (ViewGroup) findViewById(R.id.upper_header);
249         mTitleContainer = findViewById(R.id.title_container);
250         mSnapHeaderBottomBorder = findViewById(R.id.snap_header_bottom_border);
251         mSenderNameView = (TextView) findViewById(R.id.sender_name);
252         mRecipientSummary = (TextView) findViewById(R.id.recipient_summary);
253         mDateView = (TextView) findViewById(R.id.send_date);
254         mHideDetailsView = findViewById(R.id.hide_details);
255         mSnippetView = (TextView) findViewById(R.id.email_snippet);
256         mPhotoView = (MessageHeaderContactBadge) findViewById(R.id.photo);
257         mPhotoView.setQuickContactBadge(
258                 (QuickContactBadge) findViewById(R.id.invisible_quick_contact));
259         mReplyButton = findViewById(R.id.reply);
260         mReplyAllButton = findViewById(R.id.reply_all);
261         mForwardButton = findViewById(R.id.forward);
262         mOverflowButton = findViewById(R.id.overflow);
263         mDraftIcon = findViewById(R.id.draft);
264         mEditDraftButton = findViewById(R.id.edit_draft);
265         mUpperDateView = (TextView) findViewById(R.id.upper_date);
266         mAttachmentIcon = findViewById(R.id.attachment);
267         mExtraContentView = (ViewGroup) findViewById(R.id.header_extra_content);
268 
269         setExpanded(true);
270 
271         registerMessageClickTargets(mReplyButton, mReplyAllButton, mForwardButton,
272                 mEditDraftButton, mOverflowButton, mUpperHeaderView, mDateView, mHideDetailsView);
273 
274         mUpperHeaderView.setOnCreateContextMenuListener(mEmailCopyMenu);
275     }
276 
registerMessageClickTargets(View... views)277     private void registerMessageClickTargets(View... views) {
278         for (View v : views) {
279             if (v != null) {
280                 v.setOnClickListener(this);
281             }
282         }
283     }
284 
285     @Override
initialize(ConversationAccountController accountController, Map<String, Address> addressCache, MessageHeaderViewCallbacks callbacks, ContactInfoSource contactInfoSource, VeiledAddressMatcher veiledAddressMatcher)286     public void initialize(ConversationAccountController accountController,
287             Map<String, Address> addressCache, MessageHeaderViewCallbacks callbacks,
288             ContactInfoSource contactInfoSource, VeiledAddressMatcher veiledAddressMatcher) {
289         initialize(accountController, addressCache);
290         setCallbacks(callbacks);
291         setContactInfoSource(contactInfoSource);
292         setVeiledMatcher(veiledAddressMatcher);
293     }
294 
295     /**
296      * Associate the header with a contact info source for later contact
297      * presence/photo lookup.
298      */
setContactInfoSource(ContactInfoSource contactInfoSource)299     public void setContactInfoSource(ContactInfoSource contactInfoSource) {
300         mContactInfoSource = contactInfoSource;
301     }
302 
setCallbacks(MessageHeaderViewCallbacks callbacks)303     public void setCallbacks(MessageHeaderViewCallbacks callbacks) {
304         mCallbacks = callbacks;
305     }
306 
setVeiledMatcher(VeiledAddressMatcher matcher)307     public void setVeiledMatcher(VeiledAddressMatcher matcher) {
308         mVeiledMatcher = matcher;
309     }
310 
isExpanded()311     public boolean isExpanded() {
312         // (let's just arbitrarily say that unbound views are expanded by default)
313         return mMessageHeaderItem == null || mMessageHeaderItem.isExpanded();
314     }
315 
316     @Override
onDetachedFromParent()317     public void onDetachedFromParent() {
318         unbind();
319     }
320 
321     /**
322      * Headers that are unbound will not match any rendered header (matches()
323      * will return false). Unbinding is not guaranteed to *hide* the view's old
324      * data, though. To re-bind this header to message data, call render() or
325      * renderUpperHeaderFrom().
326      */
327     @Override
unbind()328     public void unbind() {
329         mMessageHeaderItem = null;
330         mMessage = null;
331 
332         if (mObservingContactInfo) {
333             mContactInfoSource.unregisterObserver(mContactInfoObserver);
334             mObservingContactInfo = false;
335         }
336     }
337 
initialize(ConversationAccountController accountController, Map<String, Address> addressCache)338     public void initialize(ConversationAccountController accountController,
339             Map<String, Address> addressCache) {
340         mAccountController = accountController;
341         mAddressCache = addressCache;
342     }
343 
getAccount()344     private Account getAccount() {
345         return mAccountController != null ? mAccountController.getAccount() : null;
346     }
347 
bind(MessageHeaderItem headerItem, boolean measureOnly)348     public void bind(MessageHeaderItem headerItem, boolean measureOnly) {
349         if (mMessageHeaderItem != null && mMessageHeaderItem == headerItem) {
350             return;
351         }
352 
353         mMessageHeaderItem = headerItem;
354         render(measureOnly);
355     }
356 
357     /**
358      * Rebinds the view to its data. This will only update the view
359      * if the {@link MessageHeaderItem} sent as a parameter is the
360      * same as the view's current {@link MessageHeaderItem} and the
361      * view's expanded state differs from the item's expanded state.
362      */
rebind(MessageHeaderItem headerItem)363     public void rebind(MessageHeaderItem headerItem) {
364         if (mMessageHeaderItem == null || mMessageHeaderItem != headerItem ||
365                 isActivated() == isExpanded()) {
366             return;
367         }
368 
369         render(false /* measureOnly */);
370     }
371 
372     @Override
refresh()373     public void refresh() {
374         render(false);
375     }
376 
getBidiFormatter()377     private BidiFormatter getBidiFormatter() {
378         if (mBidiFormatter == null) {
379             final ConversationViewAdapter adapter = mMessageHeaderItem != null
380                     ? mMessageHeaderItem.getAdapter() : null;
381             if (adapter == null) {
382                 mBidiFormatter = BidiFormatter.getInstance();
383             } else {
384                 mBidiFormatter = adapter.getBidiFormatter();
385             }
386         }
387         return mBidiFormatter;
388     }
389 
render(boolean measureOnly)390     private void render(boolean measureOnly) {
391         if (mMessageHeaderItem == null) {
392             return;
393         }
394 
395         Timer t = new Timer();
396         t.start(HEADER_RENDER_TAG);
397 
398         mRecipientSummaryValid = false;
399         mExpandedDetailsValid = false;
400 
401         mMessage = mMessageHeaderItem.getMessage();
402 
403         final Account account = getAccount();
404         final boolean alwaysShowImagesForAccount = (account != null) &&
405                 (account.settings.showImages == Settings.ShowImages.ALWAYS);
406 
407         final boolean alwaysShowImagesForMessage = mMessage.shouldShowImagePrompt();
408 
409         if (!alwaysShowImagesForMessage) {
410             // we don't need the "Show picture" prompt if the user allows images for this message
411             mShowImagePrompt = false;
412         } else if (mCallbacks.isSecure()) {
413             // in a secure view we always display the "Show picture" prompt
414             mShowImagePrompt = true;
415         } else {
416             // otherwise honor the account setting for automatically showing pictures
417             mShowImagePrompt = !alwaysShowImagesForAccount;
418         }
419 
420         setExpanded(mMessageHeaderItem.isExpanded());
421 
422         mFrom = mMessage.getFromAddresses();
423         mTo = mMessage.getToAddresses();
424         mCc = mMessage.getCcAddresses();
425         mBcc = mMessage.getBccAddresses();
426         mReplyTo = mMessage.getReplyToAddresses();
427 
428         /**
429          * Turns draft mode on or off. Draft mode hides message operations other
430          * than "edit", hides contact photo, hides presence, and changes the
431          * sender name to "Draft".
432          */
433         mIsDraft = mMessage.draftType != UIProvider.DraftType.NOT_A_DRAFT;
434         mSendingState = mMessage.sendingState;
435 
436         // If this was a sent message AND:
437         // 1. the account has a custom from, the cursor will populate the
438         // selected custom from as the fromAddress when a message is sent but
439         // not yet synced.
440         // 2. the account has no custom froms, fromAddress will be empty, and we
441         // can safely fall back and show the account name as sender since it's
442         // the only possible fromAddress.
443         String from = mMessage.getFrom();
444         if (TextUtils.isEmpty(from)) {
445             from = (account != null) ? account.getEmailAddress() : "";
446         }
447         mSender = getAddress(from);
448 
449         updateChildVisibility();
450 
451         final String snippet;
452         if (mIsDraft || mSendingState != UIProvider.ConversationSendingState.OTHER) {
453             snippet = makeSnippet(mMessage.snippet);
454         } else {
455             snippet = mMessage.snippet;
456         }
457         mSnippet = snippet == null ? null : getBidiFormatter().unicodeWrap(snippet);
458 
459         mSenderNameView.setText(getHeaderTitle());
460         setRecipientSummary();
461         setDateText();
462         mSnippetView.setText(mSnippet);
463         setAddressOnContextMenu();
464 
465         if (mUpperDateView != null) {
466             mUpperDateView.setText(mMessageHeaderItem.getTimestampShort());
467         }
468 
469         if (measureOnly) {
470             // avoid leaving any state around that would interfere with future regular bind() calls
471             unbind();
472         } else {
473             updateContactInfo();
474             if (!mObservingContactInfo) {
475                 mContactInfoSource.registerObserver(mContactInfoObserver);
476                 mObservingContactInfo = true;
477             }
478         }
479 
480         t.pause(HEADER_RENDER_TAG);
481     }
482 
483     /**
484      * Update context menu's address field for when the user long presses
485      * on the message header and attempts to copy/send email.
486      */
setAddressOnContextMenu()487     private void setAddressOnContextMenu() {
488         if (mSender != null) {
489             mEmailCopyMenu.setAddress(mSender.getAddress());
490         }
491     }
492 
493     @Override
isBoundTo(ConversationOverlayItem item)494     public boolean isBoundTo(ConversationOverlayItem item) {
495         return item == mMessageHeaderItem;
496     }
497 
getAddress(String emailStr)498     public Address getAddress(String emailStr) {
499         return Utils.getAddress(mAddressCache, emailStr);
500     }
501 
updateSpacerHeight()502     private void updateSpacerHeight() {
503         final int h = measureHeight();
504 
505         mMessageHeaderItem.setHeight(h);
506         if (mCallbacks != null) {
507             mCallbacks.setMessageSpacerHeight(mMessageHeaderItem, h);
508         }
509     }
510 
measureHeight()511     private int measureHeight() {
512         ViewGroup parent = (ViewGroup) getParent();
513         if (parent == null) {
514             LogUtils.e(LOG_TAG, new Error(), "Unable to measure height of detached header");
515             return getHeight();
516         }
517         mPreMeasuring = true;
518         final int h = Utils.measureViewHeight(this, parent);
519         mPreMeasuring = false;
520         return h;
521     }
522 
getHeaderTitle()523     private CharSequence getHeaderTitle() {
524         CharSequence title;
525         switch (mSendingState) {
526             case UIProvider.ConversationSendingState.QUEUED:
527             case UIProvider.ConversationSendingState.SENDING:
528             case UIProvider.ConversationSendingState.RETRYING:
529                 title = getResources().getString(R.string.sending);
530                 break;
531             case UIProvider.ConversationSendingState.SEND_ERROR:
532                 title = getResources().getString(R.string.message_failed);
533                 break;
534             default:
535                 if (mIsDraft) {
536                     title = SendersView.getSingularDraftString(getContext());
537                 } else {
538                     title = getBidiFormatter().unicodeWrap(
539                             getSenderName(mSender));
540                 }
541         }
542 
543         return title;
544     }
545 
setRecipientSummary()546     private void setRecipientSummary() {
547         if (!mRecipientSummaryValid) {
548             if (mMessageHeaderItem.recipientSummaryText == null) {
549                 final Account account = getAccount();
550                 final String meEmailAddress = (account != null) ? account.getEmailAddress() : "";
551                 mMessageHeaderItem.recipientSummaryText = getRecipientSummaryText(getContext(),
552                         meEmailAddress, mMyName, mTo, mCc, mBcc, mAddressCache, mVeiledMatcher,
553                         getBidiFormatter());
554             }
555             mRecipientSummary.setText(mMessageHeaderItem.recipientSummaryText);
556             mRecipientSummaryValid = true;
557         }
558     }
559 
setDateText()560     private void setDateText() {
561         if (mIsSnappy) {
562             mDateView.setText(mMessageHeaderItem.getTimestampLong());
563             mDateView.setOnClickListener(null);
564         } else {
565             mDateView.setMovementMethod(LinkMovementMethod.getInstance());
566             mDateView.setText(Html.fromHtml(getResources().getString(
567                     R.string.date_and_view_details, mMessageHeaderItem.getTimestampLong())));
568             StyleUtils.stripUnderlinesAndUrl(mDateView);
569         }
570     }
571 
572     /**
573      * Return the name, if known, or just the address.
574      */
getSenderName(Address sender)575     private static String getSenderName(Address sender) {
576         if (sender == null) {
577             return "";
578         }
579         final String displayName = sender.getPersonal();
580         return TextUtils.isEmpty(displayName) ? sender.getAddress() : displayName;
581     }
582 
setChildVisibility(int visibility, View... children)583     private static void setChildVisibility(int visibility, View... children) {
584         for (View v : children) {
585             if (v != null) {
586                 v.setVisibility(visibility);
587             }
588         }
589     }
590 
setExpanded(final boolean expanded)591     private void setExpanded(final boolean expanded) {
592         // use View's 'activated' flag to store expanded state
593         // child view state lists can use this to toggle drawables
594         setActivated(expanded);
595         if (mMessageHeaderItem != null) {
596             mMessageHeaderItem.setExpanded(expanded);
597         }
598     }
599 
600     /**
601      * Update the visibility of the many child views based on expanded/collapsed
602      * and draft/normal state.
603      */
updateChildVisibility()604     private void updateChildVisibility() {
605         // Too bad this can't be done with an XML state list...
606 
607         if (mIsViewOnlyMode) {
608             setMessageDetailsVisibility(VISIBLE);
609             setChildVisibility(GONE, mSnapHeaderBottomBorder);
610 
611             setChildVisibility(GONE, mReplyButton, mReplyAllButton, mForwardButton,
612                     mOverflowButton, mDraftIcon, mEditDraftButton,
613                     mAttachmentIcon, mUpperDateView, mSnippetView);
614             setChildVisibility(VISIBLE, mPhotoView, mRecipientSummary);
615 
616             setChildMarginEnd(mTitleContainer, 0);
617         } else if (isExpanded()) {
618             int normalVis, draftVis;
619 
620             final boolean isSnappy = isSnappy();
621             setMessageDetailsVisibility((isSnappy) ? GONE : VISIBLE);
622             setChildVisibility(isSnappy ? VISIBLE : GONE, mSnapHeaderBottomBorder);
623 
624             if (mIsDraft) {
625                 normalVis = GONE;
626                 draftVis = VISIBLE;
627             } else {
628                 normalVis = VISIBLE;
629                 draftVis = GONE;
630             }
631 
632             setReplyOrReplyAllVisible();
633             setChildVisibility(normalVis, mPhotoView, mForwardButton, mOverflowButton);
634             setChildVisibility(draftVis, mDraftIcon, mEditDraftButton);
635             setChildVisibility(VISIBLE, mRecipientSummary);
636             setChildVisibility(GONE, mAttachmentIcon, mUpperDateView, mSnippetView);
637 
638             setChildMarginEnd(mTitleContainer, 0);
639         } else {
640             setMessageDetailsVisibility(GONE);
641             setChildVisibility(GONE, mSnapHeaderBottomBorder);
642             setChildVisibility(VISIBLE, mSnippetView, mUpperDateView);
643 
644             setChildVisibility(GONE, mEditDraftButton, mReplyButton, mReplyAllButton,
645                     mForwardButton, mOverflowButton, mRecipientSummary,
646                     mDateView, mHideDetailsView);
647 
648             setChildVisibility(mMessage.hasAttachments ? VISIBLE : GONE,
649                     mAttachmentIcon);
650 
651             if (mIsDraft) {
652                 setChildVisibility(VISIBLE, mDraftIcon);
653                 setChildVisibility(GONE, mPhotoView);
654             } else {
655                 setChildVisibility(GONE, mDraftIcon);
656                 setChildVisibility(VISIBLE, mPhotoView);
657             }
658 
659             setChildMarginEnd(mTitleContainer, mTitleContainerMarginEnd);
660         }
661 
662         final ConversationViewAdapter adapter = mMessageHeaderItem.getAdapter();
663         if (adapter != null) {
664             mBorderView.setVisibility(
665                     adapter.isPreviousItemSuperCollapsed(mMessageHeaderItem) ? GONE : VISIBLE);
666         } else {
667             mBorderView.setVisibility(VISIBLE);
668         }
669     }
670 
671     /**
672      * If an overflow menu is present in this header's layout, set the
673      * visibility of "Reply" and "Reply All" actions based on a user preference.
674      * Only one of those actions will be visible when an overflow is present. If
675      * no overflow is present (e.g. big phone or tablet), it's assumed we have
676      * plenty of screen real estate and can show both.
677      */
setReplyOrReplyAllVisible()678     private void setReplyOrReplyAllVisible() {
679         if (mIsDraft) {
680             setChildVisibility(GONE, mReplyButton, mReplyAllButton);
681             return;
682         } else if (mOverflowButton == null) {
683             setChildVisibility(VISIBLE, mReplyButton, mReplyAllButton);
684             return;
685         }
686 
687         final Account account = getAccount();
688         final boolean defaultReplyAll = (account != null) ? account.settings.replyBehavior
689                 == UIProvider.DefaultReplyBehavior.REPLY_ALL : false;
690         setChildVisibility(defaultReplyAll ? GONE : VISIBLE, mReplyButton);
691         setChildVisibility(defaultReplyAll ? VISIBLE : GONE, mReplyAllButton);
692     }
693 
694     @SuppressLint("NewApi")
setChildMarginEnd(View childView, int marginEnd)695     private static void setChildMarginEnd(View childView, int marginEnd) {
696         MarginLayoutParams mlp = (MarginLayoutParams) childView.getLayoutParams();
697         if (Utils.isRunningJBMR1OrLater()) {
698             mlp.setMarginEnd(marginEnd);
699         } else {
700             mlp.rightMargin = marginEnd;
701         }
702         childView.setLayoutParams(mlp);
703     }
704 
705 
706 
707     @VisibleForTesting
getRecipientSummaryText(Context context, String meEmailAddress, String myName, String[] to, String[] cc, String[] bcc, Map<String, Address> addressCache, VeiledAddressMatcher matcher, BidiFormatter bidiFormatter)708     static CharSequence getRecipientSummaryText(Context context, String meEmailAddress,
709             String myName, String[] to, String[] cc, String[] bcc,
710             Map<String, Address> addressCache, VeiledAddressMatcher matcher,
711             BidiFormatter bidiFormatter) {
712 
713         final RecipientListsBuilder builder = new RecipientListsBuilder(
714                 context, meEmailAddress, myName, addressCache, matcher, bidiFormatter);
715 
716         builder.append(to);
717         builder.append(cc);
718         builder.appendBcc(bcc);
719 
720         return builder.build();
721     }
722 
723     /**
724      * Utility class to build a list of recipient lists.
725      */
726     private static class RecipientListsBuilder {
727         private final Context mContext;
728         private final String mMeEmailAddress;
729         private final String mMyName;
730         private final StringBuilder mBuilder = new StringBuilder();
731         private final CharSequence mComma;
732         private final Map<String, Address> mAddressCache;
733         private final VeiledAddressMatcher mMatcher;
734         private final BidiFormatter mBidiFormatter;
735 
736         int mRecipientCount = 0;
737         boolean mSkipComma = true;
738 
RecipientListsBuilder(Context context, String meEmailAddress, String myName, Map<String, Address> addressCache, VeiledAddressMatcher matcher, BidiFormatter bidiFormatter)739         public RecipientListsBuilder(Context context, String meEmailAddress, String myName,
740                 Map<String, Address> addressCache, VeiledAddressMatcher matcher,
741                 BidiFormatter bidiFormatter) {
742             mContext = context;
743             mMeEmailAddress = meEmailAddress;
744             mMyName = myName;
745             mComma = mContext.getText(R.string.enumeration_comma);
746             mAddressCache = addressCache;
747             mMatcher = matcher;
748             mBidiFormatter = bidiFormatter;
749         }
750 
append(String[] recipients)751         public void append(String[] recipients) {
752             final int addLimit = SUMMARY_MAX_RECIPIENTS - mRecipientCount;
753             final boolean hasRecipients = appendRecipients(recipients, addLimit);
754             if (hasRecipients) {
755                 mRecipientCount += Math.min(addLimit, recipients.length);
756             }
757         }
758 
appendBcc(String[] recipients)759         public void appendBcc(String[] recipients) {
760             final int addLimit = SUMMARY_MAX_RECIPIENTS - mRecipientCount;
761             if (shouldAppendRecipients(recipients, addLimit)) {
762                 // add the comma before the bcc header
763                 // and then reset mSkipComma so we don't add a comma after "bcc: "
764                 if (!mSkipComma) {
765                     mBuilder.append(mComma);
766                     mSkipComma = true;
767                 }
768                 mBuilder.append(mContext.getString(R.string.bcc_header_for_recipient_summary));
769             }
770             append(recipients);
771         }
772 
773         /**
774          * Appends formatted recipients of the message to the recipient list,
775          * as long as there are recipients left to append and the maximum number
776          * of addresses limit has not been reached.
777          * @param rawAddrs The addresses to append.
778          * @param maxToCopy The maximum number of addresses to append.
779          * @return {@code true} if a recipient has been appended. {@code false}, otherwise.
780          */
appendRecipients(String[] rawAddrs, int maxToCopy)781         private boolean appendRecipients(String[] rawAddrs, int maxToCopy) {
782             if (!shouldAppendRecipients(rawAddrs, maxToCopy)) {
783                 return false;
784             }
785 
786             final int len = Math.min(maxToCopy, rawAddrs.length);
787             for (int i = 0; i < len; i++) {
788                 final Address email = Utils.getAddress(mAddressCache, rawAddrs[i]);
789                 final String emailAddress = email.getAddress();
790                 final String name;
791                 if (mMatcher != null && mMatcher.isVeiledAddress(emailAddress)) {
792                     if (TextUtils.isEmpty(email.getPersonal())) {
793                         // Let's write something more readable.
794                         name = mContext.getString(VeiledAddressMatcher.VEILED_SUMMARY_UNKNOWN);
795                     } else {
796                         name = email.getSimplifiedName();
797                     }
798                 } else {
799                     // Not a veiled address, show first part of email, or "me".
800                     name = mMeEmailAddress.equals(emailAddress) ?
801                             mMyName : email.getSimplifiedName();
802                 }
803 
804                 // duplicate TextUtils.join() logic to minimize temporary allocations
805                 if (mSkipComma) {
806                     mSkipComma = false;
807                 } else {
808                     mBuilder.append(mComma);
809                 }
810                 mBuilder.append(mBidiFormatter.unicodeWrap(name));
811             }
812 
813             return true;
814         }
815 
816         /**
817          * @param rawAddrs The addresses to append.
818          * @param maxToCopy The maximum number of addresses to append.
819          * @return {@code true} if a recipient should be appended. {@code false}, otherwise.
820          */
shouldAppendRecipients(String[] rawAddrs, int maxToCopy)821         private boolean shouldAppendRecipients(String[] rawAddrs, int maxToCopy) {
822             return rawAddrs != null && rawAddrs.length != 0 && maxToCopy != 0;
823         }
824 
build()825         public CharSequence build() {
826             return mContext.getString(R.string.to_message_header, mBuilder);
827         }
828     }
829 
updateContactInfo()830     private void updateContactInfo() {
831         if (mContactInfoSource == null || mSender == null) {
832             mPhotoView.setImageToDefault();
833             mPhotoView.setContentDescription(getResources().getString(
834                     R.string.contact_info_string_default));
835             return;
836         }
837 
838         // Set the photo to either a found Bitmap or the default
839         // and ensure either the contact URI or email is set so the click
840         // handling works
841         String contentDesc = getResources().getString(R.string.contact_info_string,
842                 !TextUtils.isEmpty(mSender.getPersonal())
843                         ? mSender.getPersonal()
844                         : mSender.getAddress());
845         mPhotoView.setContentDescription(contentDesc);
846         boolean photoSet = false;
847         final String email = mSender.getAddress();
848         final ContactInfo info = mContactInfoSource.getContactInfo(email);
849         if (info != null) {
850             if (info.contactUri != null) {
851                 mPhotoView.assignContactUri(info.contactUri);
852             } else {
853                 mPhotoView.assignContactFromEmail(email, true /* lazyLookup */);
854             }
855 
856             if (info.photo != null) {
857                 mPhotoView.setImageBitmap(BitmapUtil.frameBitmapInCircle(info.photo));
858                 photoSet = true;
859             }
860         } else {
861             mPhotoView.assignContactFromEmail(email, true /* lazyLookup */);
862         }
863 
864         if (!photoSet) {
865             mPhotoView.setImageBitmap(
866                     BitmapUtil.frameBitmapInCircle(makeLetterTile(mSender.getPersonal(), email)));
867         }
868     }
869 
makeLetterTile( String displayName, String senderAddress)870     private Bitmap makeLetterTile(
871             String displayName, String senderAddress) {
872         if (mLetterTileProvider == null) {
873             mLetterTileProvider = new LetterTileProvider(getContext().getResources());
874         }
875 
876         final ImageCanvas.Dimensions dimensions = new ImageCanvas.Dimensions(
877                 mContactPhotoWidth, mContactPhotoHeight, ImageCanvas.Dimensions.SCALE_ONE);
878         return mLetterTileProvider.getLetterTile(dimensions, displayName, senderAddress);
879     }
880 
881     @Override
onMenuItemClick(MenuItem item)882     public boolean onMenuItemClick(MenuItem item) {
883         mPopup.dismiss();
884         return onClick(null, item.getItemId());
885     }
886 
887     @Override
onClick(View v)888     public void onClick(View v) {
889         onClick(v, v.getId());
890     }
891 
892     /**
893      * Handles clicks on either views or menu items. View parameter can be null
894      * for menu item clicks.
895      */
onClick(final View v, final int id)896     public boolean onClick(final View v, final int id) {
897         if (mMessage == null) {
898             LogUtils.i(LOG_TAG, "ignoring message header tap on unbound view");
899             return false;
900         }
901 
902         boolean handled = true;
903 
904         if (id == R.id.reply) {
905             ComposeActivity.reply(getContext(), getAccount(), mMessage);
906         } else if (id == R.id.reply_all) {
907             ComposeActivity.replyAll(getContext(), getAccount(), mMessage);
908         } else if (id == R.id.forward) {
909             ComposeActivity.forward(getContext(), getAccount(), mMessage);
910         } else if (id == R.id.add_star) {
911             mMessage.star(true);
912         } else if (id == R.id.remove_star) {
913             mMessage.star(false);
914         } else if (id == R.id.print_message) {
915             printMessage();
916         } else if (id == R.id.report_rendering_problem) {
917             final String text = getContext().getString(R.string.report_rendering_problem_desc);
918             ComposeActivity.reportRenderingFeedback(getContext(), getAccount(), mMessage,
919                     text + "\n\n" + mCallbacks.getMessageTransforms(mMessage));
920         } else if (id == R.id.report_rendering_improvement) {
921             final String text = getContext().getString(R.string.report_rendering_improvement_desc);
922             ComposeActivity.reportRenderingFeedback(getContext(), getAccount(), mMessage,
923                     text + "\n\n" + mCallbacks.getMessageTransforms(mMessage));
924         } else if (id == R.id.edit_draft) {
925             ComposeActivity.editDraft(getContext(), getAccount(), mMessage);
926         } else if (id == R.id.overflow) {
927             if (mPopup == null) {
928                 mPopup = new PopupMenu(getContext(), v);
929                 mPopup.getMenuInflater().inflate(R.menu.message_header_overflow_menu,
930                         mPopup.getMenu());
931                 mPopup.setOnMenuItemClickListener(this);
932             }
933             final boolean defaultReplyAll = getAccount().settings.replyBehavior
934                     == UIProvider.DefaultReplyBehavior.REPLY_ALL;
935             final Menu m = mPopup.getMenu();
936             m.findItem(R.id.reply).setVisible(defaultReplyAll);
937             m.findItem(R.id.reply_all).setVisible(!defaultReplyAll);
938             m.findItem(R.id.print_message).setVisible(Utils.isRunningKitkatOrLater());
939 
940             final boolean isStarred = mMessage.starred;
941             boolean showStar = true;
942             final Conversation conversation = mMessage.getConversation();
943             if (conversation != null) {
944                 showStar = !conversation.isInTrash();
945             }
946             m.findItem(R.id.add_star).setVisible(showStar && !isStarred);
947             m.findItem(R.id.remove_star).setVisible(showStar && isStarred);
948 
949             final boolean reportRendering = ENABLE_REPORT_RENDERING_PROBLEM
950                 && mCallbacks.supportsMessageTransforms();
951             m.findItem(R.id.report_rendering_improvement).setVisible(reportRendering);
952             m.findItem(R.id.report_rendering_problem).setVisible(reportRendering);
953 
954             mPopup.show();
955         } else if (id == R.id.send_date || id == R.id.hide_details ||
956                 id == R.id.details_expanded_content) {
957             toggleMessageDetails();
958         } else if (id == R.id.upper_header) {
959             toggleExpanded();
960         } else if (id == R.id.show_pictures_text) {
961             handleShowImagePromptClick(v);
962         } else {
963             LogUtils.i(LOG_TAG, "unrecognized header tap: %d", id);
964             handled = false;
965         }
966 
967         if (handled && id != R.id.overflow) {
968             Analytics.getInstance().sendMenuItemEvent(Analytics.EVENT_CATEGORY_MENU_ITEM, id,
969                     "message_header", 0);
970         }
971 
972         return handled;
973     }
974 
printMessage()975     private void printMessage() {
976         // Secure conversation view does not use a conversation view adapter
977         // so it's safe to test for existence as a signal to use javascript or not.
978         final boolean useJavascript = mMessageHeaderItem.getAdapter() != null;
979         final Account account = getAccount();
980         final Conversation conversation = mMessage.getConversation();
981         final String baseUri =
982                 AbstractConversationViewFragment.buildBaseUri(getContext(), account, conversation);
983         PrintUtils.printMessage(getContext(), mMessage, conversation.subject,
984                 mAddressCache, conversation.getBaseUri(baseUri), useJavascript);
985     }
986 
987     /**
988      * Set to true if the user should not be able to perform message actions
989      * on the message such as reply/reply all/forward/star/etc.
990      *
991      * Default is false.
992      */
setViewOnlyMode(boolean isViewOnlyMode)993     public void setViewOnlyMode(boolean isViewOnlyMode) {
994         mIsViewOnlyMode = isViewOnlyMode;
995     }
996 
setExpandable(boolean expandable)997     public void setExpandable(boolean expandable) {
998         mExpandable = expandable;
999     }
1000 
toggleExpanded()1001     public void toggleExpanded() {
1002         if (!mExpandable) {
1003             return;
1004         }
1005         setExpanded(!isExpanded());
1006 
1007         // The snappy header will disappear; no reason to update text.
1008         if (!isSnappy()) {
1009             mSenderNameView.setText(getHeaderTitle());
1010             setRecipientSummary();
1011             setDateText();
1012             mSnippetView.setText(mSnippet);
1013         }
1014 
1015         updateChildVisibility();
1016 
1017         // Force-measure the new header height so we can set the spacer size and
1018         // reveal the message div in one pass. Force-measuring makes it unnecessary to set
1019         // mSizeChanged.
1020         int h = measureHeight();
1021         mMessageHeaderItem.setHeight(h);
1022         if (mCallbacks != null) {
1023             mCallbacks.setMessageExpanded(mMessageHeaderItem, h);
1024         }
1025     }
1026 
isValidPosition(int position, int size)1027     private static boolean isValidPosition(int position, int size) {
1028         return position >= 0 && position < size;
1029     }
1030 
1031     @Override
setSnappy()1032     public void setSnappy() {
1033         mIsSnappy = true;
1034         hideMessageDetails();
1035     }
1036 
isSnappy()1037     private boolean isSnappy() {
1038         return mIsSnappy;
1039     }
1040 
toggleMessageDetails()1041     private void toggleMessageDetails() {
1042         int heightBefore = measureHeight();
1043         final boolean expand =
1044                 (mExpandedDetailsView == null || mExpandedDetailsView.getVisibility() == GONE);
1045         setMessageDetailsExpanded(expand);
1046         updateSpacerHeight();
1047         if (mCallbacks != null) {
1048             mCallbacks.setMessageDetailsExpanded(mMessageHeaderItem, expand, heightBefore);
1049         }
1050     }
1051 
setMessageDetailsExpanded(boolean expand)1052     private void setMessageDetailsExpanded(boolean expand) {
1053         if (expand) {
1054             showExpandedDetails();
1055         } else {
1056             hideExpandedDetails();
1057         }
1058 
1059         if (mMessageHeaderItem != null) {
1060             mMessageHeaderItem.detailsExpanded = expand;
1061         }
1062     }
1063 
setMessageDetailsVisibility(int vis)1064     public void setMessageDetailsVisibility(int vis) {
1065         if (vis == GONE) {
1066             hideExpandedDetails();
1067             hideSpamWarning();
1068             hideShowImagePrompt();
1069             hideInvite();
1070             mUpperHeaderView.setOnCreateContextMenuListener(null);
1071         } else {
1072             setMessageDetailsExpanded(mMessageHeaderItem.detailsExpanded);
1073             if (mMessage.spamWarningString == null) {
1074                 hideSpamWarning();
1075             } else {
1076                 showSpamWarning();
1077             }
1078             if (mShowImagePrompt) {
1079                 if (mMessageHeaderItem.getShowImages()) {
1080                     showImagePromptAlways(true);
1081                 } else {
1082                     showImagePromptOnce();
1083                 }
1084             } else {
1085                 hideShowImagePrompt();
1086             }
1087             if (mMessage.isFlaggedCalendarInvite()) {
1088                 showInvite();
1089             } else {
1090                 hideInvite();
1091             }
1092             mUpperHeaderView.setOnCreateContextMenuListener(mEmailCopyMenu);
1093         }
1094     }
1095 
hideMessageDetails()1096     private void hideMessageDetails() {
1097         setMessageDetailsVisibility(GONE);
1098     }
1099 
hideExpandedDetails()1100     private void hideExpandedDetails() {
1101         if (mExpandedDetailsView != null) {
1102             mExpandedDetailsView.setVisibility(GONE);
1103         }
1104         mDateView.setVisibility(VISIBLE);
1105         mHideDetailsView.setVisibility(GONE);
1106     }
1107 
hideInvite()1108     private void hideInvite() {
1109         if (mInviteView != null) {
1110             mInviteView.setVisibility(GONE);
1111         }
1112     }
1113 
showInvite()1114     private void showInvite() {
1115         if (mInviteView == null) {
1116             mInviteView = (MessageInviteView) mInflater.inflate(
1117                     R.layout.conversation_message_invite, this, false);
1118             mExtraContentView.addView(mInviteView);
1119         }
1120         mInviteView.bind(mMessage);
1121         mInviteView.setVisibility(VISIBLE);
1122     }
1123 
hideShowImagePrompt()1124     private void hideShowImagePrompt() {
1125         if (mImagePromptView != null) {
1126             mImagePromptView.setVisibility(GONE);
1127         }
1128     }
1129 
showImagePromptOnce()1130     private void showImagePromptOnce() {
1131         if (mImagePromptView == null) {
1132             mImagePromptView = (TextView) mInflater.inflate(
1133                     R.layout.conversation_message_show_pics, this, false);
1134             mExtraContentView.addView(mImagePromptView);
1135             mImagePromptView.setOnClickListener(this);
1136         }
1137         mImagePromptView.setVisibility(VISIBLE);
1138         mImagePromptView.setText(R.string.show_images);
1139         mImagePromptView.setTag(SHOW_IMAGE_PROMPT_ONCE);
1140     }
1141 
1142     /**
1143      * Shows the "Always show pictures" message
1144      *
1145      * @param initialShowing <code>true</code> if this is the first time we are showing the prompt
1146      *        for "show images", <code>false</code> if we are transitioning from "Show pictures"
1147      */
showImagePromptAlways(final boolean initialShowing)1148     private void showImagePromptAlways(final boolean initialShowing) {
1149         if (initialShowing) {
1150             // Initialize the view
1151             showImagePromptOnce();
1152         }
1153 
1154         mImagePromptView.setText(R.string.always_show_images);
1155         mImagePromptView.setTag(SHOW_IMAGE_PROMPT_ALWAYS);
1156 
1157         if (!initialShowing) {
1158             // the new text's line count may differ, so update the spacer height
1159             updateSpacerHeight();
1160         }
1161     }
1162 
hideSpamWarning()1163     private void hideSpamWarning() {
1164         if (mSpamWarningView != null) {
1165             mSpamWarningView.setVisibility(GONE);
1166         }
1167     }
1168 
showSpamWarning()1169     private void showSpamWarning() {
1170         if (mSpamWarningView == null) {
1171             mSpamWarningView = (SpamWarningView)
1172                     mInflater.inflate(R.layout.conversation_message_spam_warning, this, false);
1173             mExtraContentView.addView(mSpamWarningView);
1174         }
1175 
1176         mSpamWarningView.showSpamWarning(mMessage, mSender);
1177     }
1178 
handleShowImagePromptClick(View v)1179     private void handleShowImagePromptClick(View v) {
1180         Integer state = (Integer) v.getTag();
1181         if (state == null) {
1182             return;
1183         }
1184         switch (state) {
1185             case SHOW_IMAGE_PROMPT_ONCE:
1186                 if (mCallbacks != null) {
1187                     mCallbacks.showExternalResources(mMessage);
1188                 }
1189                 if (mMessageHeaderItem != null) {
1190                     mMessageHeaderItem.setShowImages(true);
1191                 }
1192                 if (mIsViewOnlyMode) {
1193                     hideShowImagePrompt();
1194                 } else {
1195                     showImagePromptAlways(false);
1196                 }
1197                 break;
1198             case SHOW_IMAGE_PROMPT_ALWAYS:
1199                 mMessage.markAlwaysShowImages(getQueryHandler(), 0 /* token */, null /* cookie */);
1200 
1201                 if (mCallbacks != null) {
1202                     mCallbacks.showExternalResources(mMessage.getFrom());
1203                 }
1204 
1205                 mShowImagePrompt = false;
1206                 v.setTag(null);
1207                 v.setVisibility(GONE);
1208                 updateSpacerHeight();
1209                 Toast.makeText(getContext(), R.string.always_show_images_toast, Toast.LENGTH_SHORT)
1210                         .show();
1211                 break;
1212         }
1213     }
1214 
getQueryHandler()1215     private AsyncQueryHandler getQueryHandler() {
1216         if (mQueryHandler == null) {
1217             mQueryHandler = new AsyncQueryHandler(getContext().getContentResolver()) {};
1218         }
1219         return mQueryHandler;
1220     }
1221 
1222     /**
1223      * Makes expanded details visible. If necessary, will inflate expanded
1224      * details layout and render using saved-off state (senders, timestamp,
1225      * etc).
1226      */
showExpandedDetails()1227     private void showExpandedDetails() {
1228         // lazily create expanded details view
1229         final boolean expandedViewCreated = ensureExpandedDetailsView();
1230         if (expandedViewCreated) {
1231             mExtraContentView.addView(mExpandedDetailsView, 0);
1232         }
1233         mExpandedDetailsView.setVisibility(VISIBLE);
1234         mDateView.setVisibility(GONE);
1235         mHideDetailsView.setVisibility(VISIBLE);
1236     }
1237 
ensureExpandedDetailsView()1238     private boolean ensureExpandedDetailsView() {
1239         boolean viewCreated = false;
1240         if (mExpandedDetailsView == null) {
1241             View v = inflateExpandedDetails(mInflater);
1242             v.setOnClickListener(this);
1243 
1244             mExpandedDetailsView = (ViewGroup) v;
1245             viewCreated = true;
1246         }
1247         if (!mExpandedDetailsValid) {
1248             renderExpandedDetails(getResources(), mExpandedDetailsView, mMessage.viaDomain,
1249                     mAddressCache, getAccount(), mVeiledMatcher, mFrom, mReplyTo, mTo, mCc, mBcc,
1250                     mMessageHeaderItem.getTimestampFull(),
1251                     getBidiFormatter());
1252 
1253             mExpandedDetailsValid = true;
1254         }
1255         return viewCreated;
1256     }
1257 
inflateExpandedDetails(LayoutInflater inflater)1258     public static View inflateExpandedDetails(LayoutInflater inflater) {
1259         return inflater.inflate(R.layout.conversation_message_header_details, null, false);
1260     }
1261 
renderExpandedDetails(Resources res, View detailsView, String viaDomain, Map<String, Address> addressCache, Account account, VeiledAddressMatcher veiledMatcher, String[] from, String[] replyTo, String[] to, String[] cc, String[] bcc, CharSequence receivedTimestamp, BidiFormatter bidiFormatter)1262     public static void renderExpandedDetails(Resources res, View detailsView,
1263             String viaDomain, Map<String, Address> addressCache, Account account,
1264             VeiledAddressMatcher veiledMatcher, String[] from, String[] replyTo,
1265             String[] to, String[] cc, String[] bcc, CharSequence receivedTimestamp,
1266             BidiFormatter bidiFormatter) {
1267         renderEmailList(res, R.id.from_heading, R.id.from_details, from, viaDomain,
1268                 detailsView, addressCache, account, veiledMatcher, bidiFormatter);
1269         renderEmailList(res, R.id.replyto_heading, R.id.replyto_details, replyTo, viaDomain,
1270                 detailsView, addressCache, account, veiledMatcher, bidiFormatter);
1271         renderEmailList(res, R.id.to_heading, R.id.to_details, to, viaDomain,
1272                 detailsView, addressCache, account, veiledMatcher, bidiFormatter);
1273         renderEmailList(res, R.id.cc_heading, R.id.cc_details, cc, viaDomain,
1274                 detailsView, addressCache, account, veiledMatcher, bidiFormatter);
1275         renderEmailList(res, R.id.bcc_heading, R.id.bcc_details, bcc, viaDomain,
1276                 detailsView, addressCache, account, veiledMatcher, bidiFormatter);
1277 
1278         // Render date
1279         detailsView.findViewById(R.id.date_heading).setVisibility(VISIBLE);
1280         final TextView date = (TextView) detailsView.findViewById(R.id.date_details);
1281         date.setText(receivedTimestamp);
1282         date.setVisibility(VISIBLE);
1283     }
1284 
1285     /**
1286      * Render an email list for the expanded message details view.
1287      */
renderEmailList(Resources res, int headerId, int detailsId, String[] emails, String viaDomain, View rootView, Map<String, Address> addressCache, Account account, VeiledAddressMatcher veiledMatcher, BidiFormatter bidiFormatter)1288     private static void renderEmailList(Resources res, int headerId, int detailsId,
1289             String[] emails, String viaDomain, View rootView,
1290             Map<String, Address> addressCache, Account account,
1291             VeiledAddressMatcher veiledMatcher, BidiFormatter bidiFormatter) {
1292         if (emails == null || emails.length == 0) {
1293             return;
1294         }
1295         final String[] formattedEmails = new String[emails.length];
1296         for (int i = 0; i < emails.length; i++) {
1297             final Address email = Utils.getAddress(addressCache, emails[i]);
1298             String name = email.getPersonal();
1299             final String address = email.getAddress();
1300             // Check if the address here is a veiled address.  If it is, we need to display an
1301             // alternate layout
1302             final boolean isVeiledAddress = veiledMatcher != null &&
1303                     veiledMatcher.isVeiledAddress(address);
1304             final String addressShown;
1305             if (isVeiledAddress) {
1306                 // Add the warning at the end of the name, and remove the address.  The alternate
1307                 // text cannot be put in the address part, because the address is made into a link,
1308                 // and the alternate human-readable text is not a link.
1309                 addressShown = "";
1310                 if (TextUtils.isEmpty(name)) {
1311                     // Empty name and we will block out the address. Let's write something more
1312                     // readable.
1313                     name = res.getString(VeiledAddressMatcher.VEILED_ALTERNATE_TEXT_UNKNOWN_PERSON);
1314                 } else {
1315                     name = name + res.getString(VeiledAddressMatcher.VEILED_ALTERNATE_TEXT);
1316                 }
1317             } else {
1318                 addressShown = address;
1319             }
1320             if (name == null || name.length() == 0 || name.equalsIgnoreCase(addressShown)) {
1321                 formattedEmails[i] = bidiFormatter.unicodeWrap(addressShown);
1322             } else {
1323                 // The one downside to having the showViaDomain here is that
1324                 // if the sender does not have a name, it will not show the via info
1325                 if (viaDomain != null) {
1326                     formattedEmails[i] = res.getString(
1327                             R.string.address_display_format_with_via_domain,
1328                             bidiFormatter.unicodeWrap(name),
1329                             bidiFormatter.unicodeWrap(addressShown),
1330                             bidiFormatter.unicodeWrap(viaDomain));
1331                 } else {
1332                     formattedEmails[i] = res.getString(R.string.address_display_format,
1333                             bidiFormatter.unicodeWrap(name),
1334                             bidiFormatter.unicodeWrap(addressShown));
1335                 }
1336             }
1337         }
1338 
1339         rootView.findViewById(headerId).setVisibility(VISIBLE);
1340         final TextView detailsText = (TextView) rootView.findViewById(detailsId);
1341         detailsText.setText(TextUtils.join("\n", formattedEmails));
1342         stripUnderlines(detailsText, account);
1343         detailsText.setVisibility(VISIBLE);
1344     }
1345 
stripUnderlines(TextView textView, Account account)1346     private static void stripUnderlines(TextView textView, Account account) {
1347         final Spannable spannable = (Spannable) textView.getText();
1348         final URLSpan[] urls = textView.getUrls();
1349 
1350         for (URLSpan span : urls) {
1351             final int start = spannable.getSpanStart(span);
1352             final int end = spannable.getSpanEnd(span);
1353             spannable.removeSpan(span);
1354             span = new EmailAddressSpan(account, span.getURL().substring(7));
1355             spannable.setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
1356         }
1357     }
1358 
1359     /**
1360      * Returns a short plaintext snippet generated from the given HTML message
1361      * body. Collapses whitespace, ignores '&lt;' and '&gt;' characters and
1362      * everything in between, and truncates the snippet to no more than 100
1363      * characters.
1364      *
1365      * @return Short plaintext snippet
1366      */
1367     @VisibleForTesting
makeSnippet(final String messageBody)1368     static String makeSnippet(final String messageBody) {
1369         if (TextUtils.isEmpty(messageBody)) {
1370             return null;
1371         }
1372 
1373         final StringBuilder snippet = new StringBuilder(MAX_SNIPPET_LENGTH);
1374 
1375         final StringReader reader = new StringReader(messageBody);
1376         try {
1377             int c;
1378             while ((c = reader.read()) != -1 && snippet.length() < MAX_SNIPPET_LENGTH) {
1379                 // Collapse whitespace.
1380                 if (Character.isWhitespace(c)) {
1381                     snippet.append(' ');
1382                     do {
1383                         c = reader.read();
1384                     } while (Character.isWhitespace(c));
1385                     if (c == -1) {
1386                         break;
1387                     }
1388                 }
1389 
1390                 if (c == '<') {
1391                     // Ignore everything up to and including the next '>'
1392                     // character.
1393                     while ((c = reader.read()) != -1) {
1394                         if (c == '>') {
1395                             break;
1396                         }
1397                     }
1398 
1399                     // If we reached the end of the message body, exit.
1400                     if (c == -1) {
1401                         break;
1402                     }
1403                 } else if (c == '&') {
1404                     // Read HTML entity.
1405                     StringBuilder sb = new StringBuilder();
1406 
1407                     while ((c = reader.read()) != -1) {
1408                         if (c == ';') {
1409                             break;
1410                         }
1411                         sb.append((char) c);
1412                     }
1413 
1414                     String entity = sb.toString();
1415                     if ("nbsp".equals(entity)) {
1416                         snippet.append(' ');
1417                     } else if ("lt".equals(entity)) {
1418                         snippet.append('<');
1419                     } else if ("gt".equals(entity)) {
1420                         snippet.append('>');
1421                     } else if ("amp".equals(entity)) {
1422                         snippet.append('&');
1423                     } else if ("quot".equals(entity)) {
1424                         snippet.append('"');
1425                     } else if ("apos".equals(entity) || "#39".equals(entity)) {
1426                         snippet.append('\'');
1427                     } else {
1428                         // Unknown entity; just append the literal string.
1429                         snippet.append('&').append(entity);
1430                         if (c == ';') {
1431                             snippet.append(';');
1432                         }
1433                     }
1434 
1435                     // If we reached the end of the message body, exit.
1436                     if (c == -1) {
1437                         break;
1438                     }
1439                 } else {
1440                     // The current character is a non-whitespace character that
1441                     // isn't inside some
1442                     // HTML tag and is not part of an HTML entity.
1443                     snippet.append((char) c);
1444                 }
1445             }
1446         } catch (IOException e) {
1447             LogUtils.wtf(LOG_TAG, e, "Really? IOException while reading a freaking string?!? ");
1448         }
1449 
1450         return snippet.toString();
1451     }
1452 
1453     @Override
onLayout(boolean changed, int l, int t, int r, int b)1454     protected void onLayout(boolean changed, int l, int t, int r, int b) {
1455         Timer perf = new Timer();
1456         perf.start(LAYOUT_TAG);
1457         super.onLayout(changed, l, t, r, b);
1458         perf.pause(LAYOUT_TAG);
1459     }
1460 
1461     @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)1462     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
1463         Timer t = new Timer();
1464         if (Timer.ENABLE_TIMER && !mPreMeasuring) {
1465             t.count("header measure id=" + mMessage.id);
1466             t.start(MEASURE_TAG);
1467         }
1468         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
1469         if (!mPreMeasuring) {
1470             t.pause(MEASURE_TAG);
1471         }
1472     }
1473 }
1474