1 /* 2 * Copyright (C) 2014 The Android Open Source Project 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 package com.android.contacts.quickcontact; 17 18 import android.animation.Animator; 19 import android.animation.Animator.AnimatorListener; 20 import android.animation.AnimatorSet; 21 import android.animation.ObjectAnimator; 22 import android.app.Activity; 23 import android.content.Context; 24 import android.content.Intent; 25 import android.content.res.Resources; 26 import android.graphics.ColorFilter; 27 import android.graphics.Rect; 28 import android.graphics.drawable.Drawable; 29 import android.os.Bundle; 30 import android.support.v7.widget.CardView; 31 import android.text.Spannable; 32 import android.text.TextUtils; 33 import android.transition.ChangeBounds; 34 import android.transition.Fade; 35 import android.transition.Transition; 36 import android.transition.Transition.TransitionListener; 37 import android.transition.TransitionManager; 38 import android.transition.TransitionSet; 39 import android.util.AttributeSet; 40 import android.util.Log; 41 import android.util.Property; 42 import android.view.ContextMenu.ContextMenuInfo; 43 import android.view.LayoutInflater; 44 import android.view.MotionEvent; 45 import android.view.View; 46 import android.view.ViewConfiguration; 47 import android.view.ViewGroup; 48 import android.widget.ImageView; 49 import android.widget.LinearLayout; 50 import android.widget.LinearLayout.LayoutParams; 51 import android.widget.RelativeLayout; 52 import android.widget.TextView; 53 54 import com.android.contacts.R; 55 import com.android.contacts.common.dialog.CallSubjectDialog; 56 57 import java.util.ArrayList; 58 import java.util.List; 59 60 /** 61 * Display entries in a LinearLayout that can be expanded to show all entries. 62 */ 63 public class ExpandingEntryCardView extends CardView { 64 65 private static final String TAG = "ExpandingEntryCardView"; 66 private static final int DURATION_EXPAND_ANIMATION_FADE_IN = 200; 67 private static final int DURATION_COLLAPSE_ANIMATION_FADE_OUT = 75; 68 private static final int DELAY_EXPAND_ANIMATION_FADE_IN = 100; 69 70 public static final int DURATION_EXPAND_ANIMATION_CHANGE_BOUNDS = 300; 71 public static final int DURATION_COLLAPSE_ANIMATION_CHANGE_BOUNDS = 300; 72 73 private static final Property<View, Integer> VIEW_LAYOUT_HEIGHT_PROPERTY = 74 new Property<View, Integer>(Integer.class, "height") { 75 @Override 76 public void set(View view, Integer height) { 77 LinearLayout.LayoutParams params = (LinearLayout.LayoutParams) 78 view.getLayoutParams(); 79 params.height = height; 80 view.setLayoutParams(params); 81 } 82 83 @Override 84 public Integer get(View view) { 85 return view.getLayoutParams().height; 86 } 87 }; 88 89 /** 90 * Entry data. 91 */ 92 public static final class Entry { 93 // No action when clicking a button is specified. 94 public static final int ACTION_NONE = 1; 95 // Button action is an intent. 96 public static final int ACTION_INTENT = 2; 97 // Button action will open the call with subject dialog. 98 public static final int ACTION_CALL_WITH_SUBJECT = 3; 99 100 private final int mId; 101 private final Drawable mIcon; 102 private final String mHeader; 103 private final String mSubHeader; 104 private final Drawable mSubHeaderIcon; 105 private final String mText; 106 private final Drawable mTextIcon; 107 private Spannable mPrimaryContentDescription; 108 private final Intent mIntent; 109 private final Drawable mAlternateIcon; 110 private final Intent mAlternateIntent; 111 private final String mAlternateContentDescription; 112 private final boolean mShouldApplyColor; 113 private final boolean mIsEditable; 114 private final EntryContextMenuInfo mEntryContextMenuInfo; 115 private final Drawable mThirdIcon; 116 private final Intent mThirdIntent; 117 private final String mThirdContentDescription; 118 private final int mIconResourceId; 119 private final int mThirdAction; 120 private final Bundle mThirdExtras; 121 Entry(int id, Drawable mainIcon, String header, String subHeader, Drawable subHeaderIcon, String text, Drawable textIcon, Spannable primaryContentDescription, Intent intent, Drawable alternateIcon, Intent alternateIntent, String alternateContentDescription, boolean shouldApplyColor, boolean isEditable, EntryContextMenuInfo entryContextMenuInfo, Drawable thirdIcon, Intent thirdIntent, String thirdContentDescription, int thirdAction, Bundle thirdExtras, int iconResourceId)122 public Entry(int id, Drawable mainIcon, String header, String subHeader, 123 Drawable subHeaderIcon, String text, Drawable textIcon, 124 Spannable primaryContentDescription, Intent intent, 125 Drawable alternateIcon, Intent alternateIntent, String alternateContentDescription, 126 boolean shouldApplyColor, boolean isEditable, 127 EntryContextMenuInfo entryContextMenuInfo, Drawable thirdIcon, Intent thirdIntent, 128 String thirdContentDescription, int thirdAction, Bundle thirdExtras, 129 int iconResourceId) { 130 mId = id; 131 mIcon = mainIcon; 132 mHeader = header; 133 mSubHeader = subHeader; 134 mSubHeaderIcon = subHeaderIcon; 135 mText = text; 136 mTextIcon = textIcon; 137 mPrimaryContentDescription = primaryContentDescription; 138 mIntent = intent; 139 mAlternateIcon = alternateIcon; 140 mAlternateIntent = alternateIntent; 141 mAlternateContentDescription = alternateContentDescription; 142 mShouldApplyColor = shouldApplyColor; 143 mIsEditable = isEditable; 144 mEntryContextMenuInfo = entryContextMenuInfo; 145 mThirdIcon = thirdIcon; 146 mThirdIntent = thirdIntent; 147 mThirdContentDescription = thirdContentDescription; 148 mThirdAction = thirdAction; 149 mThirdExtras = thirdExtras; 150 mIconResourceId = iconResourceId; 151 } 152 getIcon()153 Drawable getIcon() { 154 return mIcon; 155 } 156 getHeader()157 String getHeader() { 158 return mHeader; 159 } 160 getSubHeader()161 String getSubHeader() { 162 return mSubHeader; 163 } 164 getSubHeaderIcon()165 Drawable getSubHeaderIcon() { 166 return mSubHeaderIcon; 167 } 168 getText()169 public String getText() { 170 return mText; 171 } 172 getTextIcon()173 Drawable getTextIcon() { 174 return mTextIcon; 175 } 176 getPrimaryContentDescription()177 Spannable getPrimaryContentDescription() { 178 return mPrimaryContentDescription; 179 } 180 getIntent()181 Intent getIntent() { 182 return mIntent; 183 } 184 getAlternateIcon()185 Drawable getAlternateIcon() { 186 return mAlternateIcon; 187 } 188 getAlternateIntent()189 Intent getAlternateIntent() { 190 return mAlternateIntent; 191 } 192 getAlternateContentDescription()193 String getAlternateContentDescription() { 194 return mAlternateContentDescription; 195 } 196 shouldApplyColor()197 boolean shouldApplyColor() { 198 return mShouldApplyColor; 199 } 200 isEditable()201 boolean isEditable() { 202 return mIsEditable; 203 } 204 getId()205 int getId() { 206 return mId; 207 } 208 getEntryContextMenuInfo()209 EntryContextMenuInfo getEntryContextMenuInfo() { 210 return mEntryContextMenuInfo; 211 } 212 getThirdIcon()213 Drawable getThirdIcon() { 214 return mThirdIcon; 215 } 216 getThirdIntent()217 Intent getThirdIntent() { 218 return mThirdIntent; 219 } 220 getThirdContentDescription()221 String getThirdContentDescription() { 222 return mThirdContentDescription; 223 } 224 getIconResourceId()225 int getIconResourceId() { 226 return mIconResourceId; 227 } 228 getThirdAction()229 public int getThirdAction() { 230 return mThirdAction; 231 } 232 getThirdExtras()233 public Bundle getThirdExtras() { 234 return mThirdExtras; 235 } 236 } 237 238 public interface ExpandingEntryCardViewListener { onCollapse(int heightDelta)239 void onCollapse(int heightDelta); onExpand()240 void onExpand(); onExpandDone()241 void onExpandDone(); 242 } 243 244 private View mExpandCollapseButton; 245 private TextView mExpandCollapseTextView; 246 private TextView mTitleTextView; 247 private CharSequence mExpandButtonText; 248 private CharSequence mCollapseButtonText; 249 private OnClickListener mOnClickListener; 250 private OnCreateContextMenuListener mOnCreateContextMenuListener; 251 private boolean mIsExpanded = false; 252 /** 253 * The max number of entries to show in a collapsed card. If there are less entries passed in, 254 * then they are all shown. 255 */ 256 private int mCollapsedEntriesCount; 257 private ExpandingEntryCardViewListener mListener; 258 private List<List<Entry>> mEntries; 259 private int mNumEntries = 0; 260 private boolean mAllEntriesInflated = false; 261 private List<List<View>> mEntryViews; 262 private LinearLayout mEntriesViewGroup; 263 private final ImageView mExpandCollapseArrow; 264 private int mThemeColor; 265 private ColorFilter mThemeColorFilter; 266 /** 267 * Whether to prioritize the first entry type. If prioritized, we should show at least two 268 * of this entry type. 269 */ 270 private boolean mShowFirstEntryTypeTwice; 271 private boolean mIsAlwaysExpanded; 272 /** The ViewGroup to run the expand/collapse animation on */ 273 private ViewGroup mAnimationViewGroup; 274 private LinearLayout mBadgeContainer; 275 private final List<ImageView> mBadges; 276 private final List<Integer> mBadgeIds; 277 private final int mDividerLineHeightPixels; 278 /** 279 * List to hold the separators. This saves us from reconstructing every expand/collapse and 280 * provides a smoother animation. 281 */ 282 private List<View> mSeparators; 283 private LinearLayout mContainer; 284 285 private final OnClickListener mExpandCollapseButtonListener = new OnClickListener() { 286 @Override 287 public void onClick(View v) { 288 if (mIsExpanded) { 289 collapse(); 290 } else { 291 expand(); 292 } 293 } 294 }; 295 ExpandingEntryCardView(Context context)296 public ExpandingEntryCardView(Context context) { 297 this(context, null); 298 } 299 ExpandingEntryCardView(Context context, AttributeSet attrs)300 public ExpandingEntryCardView(Context context, AttributeSet attrs) { 301 super(context, attrs); 302 LayoutInflater inflater = LayoutInflater.from(context); 303 View expandingEntryCardView = inflater.inflate(R.layout.expanding_entry_card_view, this); 304 mEntriesViewGroup = (LinearLayout) 305 expandingEntryCardView.findViewById(R.id.content_area_linear_layout); 306 mTitleTextView = (TextView) expandingEntryCardView.findViewById(R.id.title); 307 mContainer = (LinearLayout) expandingEntryCardView.findViewById(R.id.container); 308 309 mExpandCollapseButton = inflater.inflate( 310 R.layout.quickcontact_expanding_entry_card_button, this, false); 311 mExpandCollapseTextView = (TextView) mExpandCollapseButton.findViewById(R.id.text); 312 mExpandCollapseArrow = (ImageView) mExpandCollapseButton.findViewById(R.id.arrow); 313 mExpandCollapseButton.setOnClickListener(mExpandCollapseButtonListener); 314 mBadgeContainer = (LinearLayout) mExpandCollapseButton.findViewById(R.id.badge_container); 315 mDividerLineHeightPixels = getResources() 316 .getDimensionPixelSize(R.dimen.divider_line_height); 317 318 mBadges = new ArrayList<ImageView>(); 319 mBadgeIds = new ArrayList<Integer>(); 320 } 321 initialize(List<List<Entry>> entries, int numInitialVisibleEntries, boolean isExpanded, boolean isAlwaysExpanded, ExpandingEntryCardViewListener listener, ViewGroup animationViewGroup)322 public void initialize(List<List<Entry>> entries, int numInitialVisibleEntries, 323 boolean isExpanded, boolean isAlwaysExpanded, ExpandingEntryCardViewListener listener, 324 ViewGroup animationViewGroup) { 325 initialize(entries, numInitialVisibleEntries, isExpanded, isAlwaysExpanded, 326 listener, animationViewGroup, /* showFirstEntryTypeTwice = */ false); 327 } 328 329 /** 330 * Sets the Entry list to display. 331 * 332 * @param entries The Entry list to display. 333 */ initialize(List<List<Entry>> entries, int numInitialVisibleEntries, boolean isExpanded, boolean isAlwaysExpanded, ExpandingEntryCardViewListener listener, ViewGroup animationViewGroup, boolean showFirstEntryTypeTwice)334 public void initialize(List<List<Entry>> entries, int numInitialVisibleEntries, 335 boolean isExpanded, boolean isAlwaysExpanded, 336 ExpandingEntryCardViewListener listener, ViewGroup animationViewGroup, 337 boolean showFirstEntryTypeTwice) { 338 LayoutInflater layoutInflater = LayoutInflater.from(getContext()); 339 mIsExpanded = isExpanded; 340 mIsAlwaysExpanded = isAlwaysExpanded; 341 // If isAlwaysExpanded is true, mIsExpanded should be true 342 mIsExpanded |= mIsAlwaysExpanded; 343 mEntryViews = new ArrayList<List<View>>(entries.size()); 344 mEntries = entries; 345 mNumEntries = 0; 346 mAllEntriesInflated = false; 347 mShowFirstEntryTypeTwice = showFirstEntryTypeTwice; 348 for (List<Entry> entryList : mEntries) { 349 mNumEntries += entryList.size(); 350 mEntryViews.add(new ArrayList<View>()); 351 } 352 mCollapsedEntriesCount = Math.min(numInitialVisibleEntries, mNumEntries); 353 // We need a separator between each list, but not after the last one 354 if (entries.size() > 1) { 355 mSeparators = new ArrayList<>(entries.size() - 1); 356 } 357 mListener = listener; 358 mAnimationViewGroup = animationViewGroup; 359 360 if (mIsExpanded) { 361 updateExpandCollapseButton(getCollapseButtonText(), /* duration = */ 0); 362 inflateAllEntries(layoutInflater); 363 } else { 364 updateExpandCollapseButton(getExpandButtonText(), /* duration = */ 0); 365 inflateInitialEntries(layoutInflater); 366 } 367 insertEntriesIntoViewGroup(); 368 applyColor(); 369 } 370 371 /** 372 * Sets the text for the expand button. 373 * 374 * @param expandButtonText The expand button text. 375 */ setExpandButtonText(CharSequence expandButtonText)376 public void setExpandButtonText(CharSequence expandButtonText) { 377 mExpandButtonText = expandButtonText; 378 if (mExpandCollapseTextView != null && !mIsExpanded) { 379 mExpandCollapseTextView.setText(expandButtonText); 380 } 381 } 382 383 /** 384 * Sets the text for the expand button. 385 * 386 * @param expandButtonText The expand button text. 387 */ setCollapseButtonText(CharSequence expandButtonText)388 public void setCollapseButtonText(CharSequence expandButtonText) { 389 mCollapseButtonText = expandButtonText; 390 if (mExpandCollapseTextView != null && mIsExpanded) { 391 mExpandCollapseTextView.setText(mCollapseButtonText); 392 } 393 } 394 395 @Override setOnClickListener(OnClickListener listener)396 public void setOnClickListener(OnClickListener listener) { 397 mOnClickListener = listener; 398 } 399 400 @Override setOnCreateContextMenuListener(OnCreateContextMenuListener listener)401 public void setOnCreateContextMenuListener (OnCreateContextMenuListener listener) { 402 mOnCreateContextMenuListener = listener; 403 } 404 calculateEntriesToRemoveDuringCollapse()405 private List<View> calculateEntriesToRemoveDuringCollapse() { 406 final List<View> viewsToRemove = getViewsToDisplay(true); 407 final List<View> viewsCollapsed = getViewsToDisplay(false); 408 viewsToRemove.removeAll(viewsCollapsed); 409 return viewsToRemove; 410 } 411 insertEntriesIntoViewGroup()412 private void insertEntriesIntoViewGroup() { 413 mEntriesViewGroup.removeAllViews(); 414 415 for (View view : getViewsToDisplay(mIsExpanded)) { 416 mEntriesViewGroup.addView(view); 417 } 418 419 removeView(mExpandCollapseButton); 420 if (mCollapsedEntriesCount < mNumEntries 421 && mExpandCollapseButton.getParent() == null && !mIsAlwaysExpanded) { 422 mContainer.addView(mExpandCollapseButton, -1); 423 } 424 } 425 426 /** 427 * Returns the list of views that should be displayed. This changes depending on whether 428 * the card is expanded or collapsed. 429 */ getViewsToDisplay(boolean isExpanded)430 private List<View> getViewsToDisplay(boolean isExpanded) { 431 final List<View> viewsToDisplay = new ArrayList<View>(); 432 if (isExpanded) { 433 for (int i = 0; i < mEntryViews.size(); i++) { 434 List<View> viewList = mEntryViews.get(i); 435 if (i > 0) { 436 View separator; 437 if (mSeparators.size() <= i - 1) { 438 separator = generateSeparator(viewList.get(0)); 439 mSeparators.add(separator); 440 } else { 441 separator = mSeparators.get(i - 1); 442 } 443 viewsToDisplay.add(separator); 444 } 445 for (View view : viewList) { 446 viewsToDisplay.add(view); 447 } 448 } 449 } else { 450 // We want to insert mCollapsedEntriesCount entries into the group. extraEntries is the 451 // number of entries that need to be added that are not the head element of a list 452 // to reach mCollapsedEntriesCount. 453 int numInViewGroup = 0; 454 int extraEntries = mCollapsedEntriesCount - mEntryViews.size(); 455 for (int i = 0; i < mEntryViews.size() && numInViewGroup < mCollapsedEntriesCount; 456 i++) { 457 List<View> entryViewList = mEntryViews.get(i); 458 if (i > 0) { 459 View separator; 460 if (mSeparators.size() <= i - 1) { 461 separator = generateSeparator(entryViewList.get(0)); 462 mSeparators.add(separator); 463 } else { 464 separator = mSeparators.get(i - 1); 465 } 466 viewsToDisplay.add(separator); 467 } 468 viewsToDisplay.add(entryViewList.get(0)); 469 numInViewGroup++; 470 471 int indexInEntryViewList = 1; 472 if (mShowFirstEntryTypeTwice && i == 0 && entryViewList.size() > 1) { 473 viewsToDisplay.add(entryViewList.get(1)); 474 numInViewGroup++; 475 extraEntries--; 476 indexInEntryViewList++; 477 } 478 479 // Insert entries in this list to hit mCollapsedEntriesCount. 480 for (int j = indexInEntryViewList; 481 j < entryViewList.size() && numInViewGroup < mCollapsedEntriesCount && 482 extraEntries > 0; 483 j++) { 484 viewsToDisplay.add(entryViewList.get(j)); 485 numInViewGroup++; 486 extraEntries--; 487 } 488 } 489 } 490 491 formatEntryIfFirst(viewsToDisplay); 492 return viewsToDisplay; 493 } 494 formatEntryIfFirst(List<View> entriesViewGroup)495 private void formatEntryIfFirst(List<View> entriesViewGroup) { 496 // If no title and the first entry in the group, add extra padding 497 if (TextUtils.isEmpty(mTitleTextView.getText()) && 498 entriesViewGroup.size() > 0) { 499 final View entry = entriesViewGroup.get(0); 500 entry.setPadding(entry.getPaddingLeft(), 501 getResources().getDimensionPixelSize( 502 R.dimen.expanding_entry_card_item_padding_top) + 503 getResources().getDimensionPixelSize( 504 R.dimen.expanding_entry_card_null_title_top_extra_padding), 505 entry.getPaddingRight(), 506 entry.getPaddingBottom()); 507 } 508 } 509 generateSeparator(View entry)510 private View generateSeparator(View entry) { 511 View separator = new View(getContext()); 512 Resources res = getResources(); 513 514 separator.setBackgroundColor(res.getColor( 515 R.color.divider_line_color_light)); 516 LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams( 517 ViewGroup.LayoutParams.MATCH_PARENT, mDividerLineHeightPixels); 518 // The separator is aligned with the text in the entry. This is offset by a default 519 // margin. If there is an icon present, the icon's width and margin are added 520 int marginStart = res.getDimensionPixelSize( 521 R.dimen.expanding_entry_card_item_padding_start); 522 ImageView entryIcon = (ImageView) entry.findViewById(R.id.icon); 523 if (entryIcon.getVisibility() == View.VISIBLE) { 524 int imageWidthAndMargin = 525 res.getDimensionPixelSize(R.dimen.expanding_entry_card_item_icon_width) + 526 res.getDimensionPixelSize(R.dimen.expanding_entry_card_item_image_spacing); 527 marginStart += imageWidthAndMargin; 528 } 529 layoutParams.setMarginStart(marginStart); 530 separator.setLayoutParams(layoutParams); 531 return separator; 532 } 533 getExpandButtonText()534 private CharSequence getExpandButtonText() { 535 if (!TextUtils.isEmpty(mExpandButtonText)) { 536 return mExpandButtonText; 537 } else { 538 // Default to "See more". 539 return getResources().getText(R.string.expanding_entry_card_view_see_more); 540 } 541 } 542 getCollapseButtonText()543 private CharSequence getCollapseButtonText() { 544 if (!TextUtils.isEmpty(mCollapseButtonText)) { 545 return mCollapseButtonText; 546 } else { 547 // Default to "See less". 548 return getResources().getText(R.string.expanding_entry_card_view_see_less); 549 } 550 } 551 552 /** 553 * Inflates the initial entries to be shown. 554 */ inflateInitialEntries(LayoutInflater layoutInflater)555 private void inflateInitialEntries(LayoutInflater layoutInflater) { 556 // If the number of collapsed entries equals total entries, inflate all 557 if (mCollapsedEntriesCount == mNumEntries) { 558 inflateAllEntries(layoutInflater); 559 } else { 560 // Otherwise inflate the top entry from each list 561 // extraEntries is used to add extra entries until mCollapsedEntriesCount is reached. 562 int numInflated = 0; 563 int extraEntries = mCollapsedEntriesCount - mEntries.size(); 564 for (int i = 0; i < mEntries.size() && numInflated < mCollapsedEntriesCount; i++) { 565 List<Entry> entryList = mEntries.get(i); 566 List<View> entryViewList = mEntryViews.get(i); 567 568 entryViewList.add(createEntryView(layoutInflater, entryList.get(0), 569 /* showIcon = */ View.VISIBLE)); 570 numInflated++; 571 572 int indexInEntryViewList = 1; 573 if (mShowFirstEntryTypeTwice && i == 0 && entryList.size() > 1) { 574 entryViewList.add(createEntryView(layoutInflater, entryList.get(1), 575 /* showIcon = */ View.INVISIBLE)); 576 numInflated++; 577 extraEntries--; 578 indexInEntryViewList++; 579 } 580 581 // Inflate entries in this list to hit mCollapsedEntriesCount. 582 for (int j = indexInEntryViewList; j < entryList.size() 583 && numInflated < mCollapsedEntriesCount 584 && extraEntries > 0; j++) { 585 entryViewList.add(createEntryView(layoutInflater, entryList.get(j), 586 /* showIcon = */ View.INVISIBLE)); 587 numInflated++; 588 extraEntries--; 589 } 590 } 591 } 592 } 593 594 /** 595 * Inflates all entries. 596 */ inflateAllEntries(LayoutInflater layoutInflater)597 private void inflateAllEntries(LayoutInflater layoutInflater) { 598 if (mAllEntriesInflated) { 599 return; 600 } 601 for (int i = 0; i < mEntries.size(); i++) { 602 List<Entry> entryList = mEntries.get(i); 603 List<View> viewList = mEntryViews.get(i); 604 for (int j = viewList.size(); j < entryList.size(); j++) { 605 final int iconVisibility; 606 final Entry entry = entryList.get(j); 607 // If the entry does not have an icon, mark gone. Else if it has an icon, show 608 // for the first Entry in the list only 609 if (entry.getIcon() == null) { 610 iconVisibility = View.GONE; 611 } else if (j == 0) { 612 iconVisibility = View.VISIBLE; 613 } else { 614 iconVisibility = View.INVISIBLE; 615 } 616 viewList.add(createEntryView(layoutInflater, entry, iconVisibility)); 617 } 618 } 619 mAllEntriesInflated = true; 620 } 621 setColorAndFilter(int color, ColorFilter colorFilter)622 public void setColorAndFilter(int color, ColorFilter colorFilter) { 623 mThemeColor = color; 624 mThemeColorFilter = colorFilter; 625 applyColor(); 626 } 627 setEntryHeaderColor(int color)628 public void setEntryHeaderColor(int color) { 629 if (mEntries != null) { 630 for (List<View> entryList : mEntryViews) { 631 for (View entryView : entryList) { 632 TextView header = (TextView) entryView.findViewById(R.id.header); 633 if (header != null) { 634 header.setTextColor(color); 635 } 636 } 637 } 638 } 639 } 640 641 /** 642 * The ColorFilter is passed in along with the color so that a new one only needs to be created 643 * once for the entire activity. 644 * 1. Title 645 * 2. Entry icons 646 * 3. Expand/Collapse Text 647 * 4. Expand/Collapse Button 648 */ applyColor()649 public void applyColor() { 650 if (mThemeColor != 0 && mThemeColorFilter != null) { 651 // Title 652 if (mTitleTextView != null) { 653 mTitleTextView.setTextColor(mThemeColor); 654 } 655 656 // Entry icons 657 if (mEntries != null) { 658 for (List<Entry> entryList : mEntries) { 659 for (Entry entry : entryList) { 660 if (entry.shouldApplyColor()) { 661 Drawable icon = entry.getIcon(); 662 if (icon != null) { 663 icon.mutate(); 664 icon.setColorFilter(mThemeColorFilter); 665 } 666 } 667 Drawable alternateIcon = entry.getAlternateIcon(); 668 if (alternateIcon != null) { 669 alternateIcon.mutate(); 670 alternateIcon.setColorFilter(mThemeColorFilter); 671 } 672 Drawable thirdIcon = entry.getThirdIcon(); 673 if (thirdIcon != null) { 674 thirdIcon.mutate(); 675 thirdIcon.setColorFilter(mThemeColorFilter); 676 } 677 } 678 } 679 } 680 681 // Expand/Collapse 682 mExpandCollapseTextView.setTextColor(mThemeColor); 683 mExpandCollapseArrow.setColorFilter(mThemeColorFilter); 684 } 685 } 686 createEntryView(LayoutInflater layoutInflater, final Entry entry, int iconVisibility)687 private View createEntryView(LayoutInflater layoutInflater, final Entry entry, 688 int iconVisibility) { 689 final EntryView view = (EntryView) layoutInflater.inflate( 690 R.layout.expanding_entry_card_item, this, false); 691 692 view.setContextMenuInfo(entry.getEntryContextMenuInfo()); 693 if (!TextUtils.isEmpty(entry.getPrimaryContentDescription())) { 694 view.setContentDescription(entry.getPrimaryContentDescription()); 695 } 696 697 final ImageView icon = (ImageView) view.findViewById(R.id.icon); 698 icon.setVisibility(iconVisibility); 699 if (entry.getIcon() != null) { 700 icon.setImageDrawable(entry.getIcon()); 701 } 702 final TextView header = (TextView) view.findViewById(R.id.header); 703 if (!TextUtils.isEmpty(entry.getHeader())) { 704 header.setText(entry.getHeader()); 705 } else { 706 header.setVisibility(View.GONE); 707 } 708 709 final TextView subHeader = (TextView) view.findViewById(R.id.sub_header); 710 if (!TextUtils.isEmpty(entry.getSubHeader())) { 711 subHeader.setText(entry.getSubHeader()); 712 } else { 713 subHeader.setVisibility(View.GONE); 714 } 715 716 final ImageView subHeaderIcon = (ImageView) view.findViewById(R.id.icon_sub_header); 717 if (entry.getSubHeaderIcon() != null) { 718 subHeaderIcon.setImageDrawable(entry.getSubHeaderIcon()); 719 } else { 720 subHeaderIcon.setVisibility(View.GONE); 721 } 722 723 final TextView text = (TextView) view.findViewById(R.id.text); 724 if (!TextUtils.isEmpty(entry.getText())) { 725 text.setText(entry.getText()); 726 } else { 727 text.setVisibility(View.GONE); 728 } 729 730 final ImageView textIcon = (ImageView) view.findViewById(R.id.icon_text); 731 if (entry.getTextIcon() != null) { 732 textIcon.setImageDrawable(entry.getTextIcon()); 733 } else { 734 textIcon.setVisibility(View.GONE); 735 } 736 737 if (entry.getIntent() != null) { 738 view.setOnClickListener(mOnClickListener); 739 view.setTag(new EntryTag(entry.getId(), entry.getIntent())); 740 } 741 742 if (entry.getIntent() == null && entry.getEntryContextMenuInfo() == null) { 743 // Remove the click effect 744 view.setBackground(null); 745 } 746 747 // If only the header is visible, add a top margin to match icon's top margin. 748 // Also increase the space below the header for visual comfort. 749 if (header.getVisibility() == View.VISIBLE && subHeader.getVisibility() == View.GONE && 750 text.getVisibility() == View.GONE) { 751 RelativeLayout.LayoutParams headerLayoutParams = 752 (RelativeLayout.LayoutParams) header.getLayoutParams(); 753 headerLayoutParams.topMargin = (int) (getResources().getDimension( 754 R.dimen.expanding_entry_card_item_header_only_margin_top)); 755 headerLayoutParams.bottomMargin += (int) (getResources().getDimension( 756 R.dimen.expanding_entry_card_item_header_only_margin_bottom)); 757 header.setLayoutParams(headerLayoutParams); 758 } 759 760 // Adjust the top padding size for entries with an invisible icon. The padding depends on 761 // if there is a sub header or text section 762 if (iconVisibility == View.INVISIBLE && 763 (!TextUtils.isEmpty(entry.getSubHeader()) || !TextUtils.isEmpty(entry.getText()))) { 764 view.setPaddingRelative(view.getPaddingStart(), 765 getResources().getDimensionPixelSize( 766 R.dimen.expanding_entry_card_item_no_icon_margin_top), 767 view.getPaddingEnd(), 768 view.getPaddingBottom()); 769 } else if (iconVisibility == View.INVISIBLE && TextUtils.isEmpty(entry.getSubHeader()) 770 && TextUtils.isEmpty(entry.getText())) { 771 view.setPaddingRelative(view.getPaddingStart(), 0, view.getPaddingEnd(), 772 view.getPaddingBottom()); 773 } 774 775 final ImageView alternateIcon = (ImageView) view.findViewById(R.id.icon_alternate); 776 final ImageView thirdIcon = (ImageView) view.findViewById(R.id.third_icon); 777 778 if (entry.getAlternateIcon() != null && entry.getAlternateIntent() != null) { 779 alternateIcon.setImageDrawable(entry.getAlternateIcon()); 780 alternateIcon.setOnClickListener(mOnClickListener); 781 alternateIcon.setTag(new EntryTag(entry.getId(), entry.getAlternateIntent())); 782 alternateIcon.setVisibility(View.VISIBLE); 783 alternateIcon.setContentDescription(entry.getAlternateContentDescription()); 784 } 785 786 if (entry.getThirdIcon() != null && entry.getThirdAction() != Entry.ACTION_NONE) { 787 thirdIcon.setImageDrawable(entry.getThirdIcon()); 788 if (entry.getThirdAction() == Entry.ACTION_INTENT) { 789 thirdIcon.setOnClickListener(mOnClickListener); 790 thirdIcon.setTag(new EntryTag(entry.getId(), entry.getThirdIntent())); 791 } else if (entry.getThirdAction() == Entry.ACTION_CALL_WITH_SUBJECT) { 792 thirdIcon.setOnClickListener(new View.OnClickListener() { 793 @Override 794 public void onClick(View v) { 795 Object tag = v.getTag(); 796 if (!(tag instanceof Bundle)) { 797 return; 798 } 799 800 Context context = getContext(); 801 if (context instanceof Activity) { 802 CallSubjectDialog.start((Activity) context, entry.getThirdExtras()); 803 } 804 } 805 }); 806 thirdIcon.setTag(entry.getThirdExtras()); 807 } 808 thirdIcon.setVisibility(View.VISIBLE); 809 thirdIcon.setContentDescription(entry.getThirdContentDescription()); 810 } 811 812 // Set a custom touch listener for expanding the extra icon touch areas 813 view.setOnTouchListener(new EntryTouchListener(view, alternateIcon, thirdIcon)); 814 view.setOnCreateContextMenuListener(mOnCreateContextMenuListener); 815 816 return view; 817 } 818 updateExpandCollapseButton(CharSequence buttonText, long duration)819 private void updateExpandCollapseButton(CharSequence buttonText, long duration) { 820 if (mIsExpanded) { 821 final ObjectAnimator animator = ObjectAnimator.ofFloat(mExpandCollapseArrow, 822 "rotation", 180); 823 animator.setDuration(duration); 824 animator.start(); 825 } else { 826 final ObjectAnimator animator = ObjectAnimator.ofFloat(mExpandCollapseArrow, 827 "rotation", 0); 828 animator.setDuration(duration); 829 animator.start(); 830 } 831 updateBadges(); 832 833 mExpandCollapseTextView.setText(buttonText); 834 } 835 updateBadges()836 private void updateBadges() { 837 if (mIsExpanded) { 838 mBadgeContainer.removeAllViews(); 839 } else { 840 int numberOfMimeTypesShown = mCollapsedEntriesCount; 841 if (mShowFirstEntryTypeTwice && mEntries.size() > 0 842 && mEntries.get(0).size() > 1) { 843 numberOfMimeTypesShown--; 844 } 845 // Inflate badges if not yet created 846 if (mBadges.size() < mEntries.size() - numberOfMimeTypesShown) { 847 for (int i = numberOfMimeTypesShown; i < mEntries.size(); i++) { 848 Drawable badgeDrawable = mEntries.get(i).get(0).getIcon(); 849 int badgeResourceId = mEntries.get(i).get(0).getIconResourceId(); 850 // Do not add the same badge twice 851 if (badgeResourceId != 0 && mBadgeIds.contains(badgeResourceId)) { 852 continue; 853 } 854 if (badgeDrawable != null) { 855 ImageView badgeView = new ImageView(getContext()); 856 LinearLayout.LayoutParams badgeViewParams = new LinearLayout.LayoutParams( 857 (int) getResources().getDimension( 858 R.dimen.expanding_entry_card_item_icon_width), 859 (int) getResources().getDimension( 860 R.dimen.expanding_entry_card_item_icon_height)); 861 badgeViewParams.setMarginEnd((int) getResources().getDimension( 862 R.dimen.expanding_entry_card_badge_separator_margin)); 863 badgeView.setLayoutParams(badgeViewParams); 864 badgeView.setImageDrawable(badgeDrawable); 865 mBadges.add(badgeView); 866 mBadgeIds.add(badgeResourceId); 867 } 868 } 869 } 870 mBadgeContainer.removeAllViews(); 871 for (ImageView badge : mBadges) { 872 mBadgeContainer.addView(badge); 873 } 874 } 875 } 876 expand()877 private void expand() { 878 ChangeBounds boundsTransition = new ChangeBounds(); 879 boundsTransition.setDuration(DURATION_EXPAND_ANIMATION_CHANGE_BOUNDS); 880 881 Fade fadeIn = new Fade(Fade.IN); 882 fadeIn.setDuration(DURATION_EXPAND_ANIMATION_FADE_IN); 883 fadeIn.setStartDelay(DELAY_EXPAND_ANIMATION_FADE_IN); 884 885 TransitionSet transitionSet = new TransitionSet(); 886 transitionSet.addTransition(boundsTransition); 887 transitionSet.addTransition(fadeIn); 888 889 transitionSet.excludeTarget(R.id.text, /* exclude = */ true); 890 891 final ViewGroup transitionViewContainer = mAnimationViewGroup == null ? 892 this : mAnimationViewGroup; 893 894 transitionSet.addListener(new TransitionListener() { 895 @Override 896 public void onTransitionStart(Transition transition) { 897 mListener.onExpand(); 898 } 899 900 @Override 901 public void onTransitionEnd(Transition transition) { 902 mListener.onExpandDone(); 903 } 904 905 @Override 906 public void onTransitionCancel(Transition transition) { 907 } 908 909 @Override 910 public void onTransitionPause(Transition transition) { 911 } 912 913 @Override 914 public void onTransitionResume(Transition transition) { 915 } 916 }); 917 918 TransitionManager.beginDelayedTransition(transitionViewContainer, transitionSet); 919 920 mIsExpanded = true; 921 // In order to insert new entries, we may need to inflate them for the first time 922 inflateAllEntries(LayoutInflater.from(getContext())); 923 insertEntriesIntoViewGroup(); 924 updateExpandCollapseButton(getCollapseButtonText(), 925 DURATION_EXPAND_ANIMATION_CHANGE_BOUNDS); 926 } 927 collapse()928 private void collapse() { 929 final List<View> views = calculateEntriesToRemoveDuringCollapse(); 930 931 // This animation requires layout changes, unlike the expand() animation: the action bar 932 // might get scrolled open in order to fill empty space. As a result, we can't use 933 // ChangeBounds here. Instead manually animate view height and alpha. This isn't as 934 // efficient as the bounds and translation changes performed by ChangeBounds. Nonetheless, a 935 // reasonable frame-rate is achieved collapsing a dozen elements on a user Svelte N4. So the 936 // performance hit doesn't justify writing a less maintainable animation. 937 final AnimatorSet set = new AnimatorSet(); 938 final List<Animator> animators = new ArrayList<Animator>(views.size()); 939 int totalSizeChange = 0; 940 for (View viewToRemove : views) { 941 final ObjectAnimator animator = ObjectAnimator.ofObject(viewToRemove, 942 VIEW_LAYOUT_HEIGHT_PROPERTY, null, viewToRemove.getHeight(), 0); 943 totalSizeChange += viewToRemove.getHeight(); 944 animator.setDuration(DURATION_COLLAPSE_ANIMATION_CHANGE_BOUNDS); 945 animators.add(animator); 946 viewToRemove.animate().alpha(0).setDuration(DURATION_COLLAPSE_ANIMATION_FADE_OUT); 947 } 948 set.playTogether(animators); 949 set.start(); 950 set.addListener(new AnimatorListener() { 951 @Override 952 public void onAnimationStart(Animator animation) { 953 } 954 955 @Override 956 public void onAnimationEnd(Animator animation) { 957 // Now that the views have been animated away, actually remove them from the view 958 // hierarchy. Reset their appearance so that they look appropriate when they 959 // get added back later. 960 insertEntriesIntoViewGroup(); 961 for (View view : views) { 962 if (view instanceof EntryView) { 963 VIEW_LAYOUT_HEIGHT_PROPERTY.set(view, LayoutParams.WRAP_CONTENT); 964 } else { 965 VIEW_LAYOUT_HEIGHT_PROPERTY.set(view, mDividerLineHeightPixels); 966 } 967 view.animate().cancel(); 968 view.setAlpha(1); 969 } 970 } 971 972 @Override 973 public void onAnimationCancel(Animator animation) { 974 } 975 976 @Override 977 public void onAnimationRepeat(Animator animation) { 978 } 979 }); 980 981 mListener.onCollapse(totalSizeChange); 982 mIsExpanded = false; 983 updateExpandCollapseButton(getExpandButtonText(), 984 DURATION_COLLAPSE_ANIMATION_CHANGE_BOUNDS); 985 } 986 987 /** 988 * Returns whether the view is currently in its expanded state. 989 */ isExpanded()990 public boolean isExpanded() { 991 return mIsExpanded; 992 } 993 994 /** 995 * Sets the title text of this ExpandingEntryCardView. 996 * @param title The title to set. A null title will result in the title being removed. 997 */ setTitle(String title)998 public void setTitle(String title) { 999 if (mTitleTextView == null) { 1000 Log.e(TAG, "mTitleTextView is null"); 1001 } 1002 mTitleTextView.setText(title); 1003 mTitleTextView.setVisibility(TextUtils.isEmpty(title) ? View.GONE : View.VISIBLE); 1004 findViewById(R.id.title_separator).setVisibility(TextUtils.isEmpty(title) ? 1005 View.GONE : View.VISIBLE); 1006 // If the title is set after children have been added, reset the top entry's padding to 1007 // the default. Else if the title is cleared after children have been added, set 1008 // the extra top padding 1009 if (!TextUtils.isEmpty(title) && mEntriesViewGroup.getChildCount() > 0) { 1010 View firstEntry = mEntriesViewGroup.getChildAt(0); 1011 firstEntry.setPadding(firstEntry.getPaddingLeft(), 1012 getResources().getDimensionPixelSize( 1013 R.dimen.expanding_entry_card_item_padding_top), 1014 firstEntry.getPaddingRight(), 1015 firstEntry.getPaddingBottom()); 1016 } else if (!TextUtils.isEmpty(title) && mEntriesViewGroup.getChildCount() > 0) { 1017 View firstEntry = mEntriesViewGroup.getChildAt(0); 1018 firstEntry.setPadding(firstEntry.getPaddingLeft(), 1019 getResources().getDimensionPixelSize( 1020 R.dimen.expanding_entry_card_item_padding_top) + 1021 getResources().getDimensionPixelSize( 1022 R.dimen.expanding_entry_card_null_title_top_extra_padding), 1023 firstEntry.getPaddingRight(), 1024 firstEntry.getPaddingBottom()); 1025 } 1026 } 1027 shouldShow()1028 public boolean shouldShow() { 1029 return mEntries != null && mEntries.size() > 0; 1030 } 1031 1032 public static final class EntryView extends RelativeLayout { 1033 private EntryContextMenuInfo mEntryContextMenuInfo; 1034 EntryView(Context context)1035 public EntryView(Context context) { 1036 super(context); 1037 } 1038 EntryView(Context context, AttributeSet attrs)1039 public EntryView(Context context, AttributeSet attrs) { 1040 super(context, attrs); 1041 } 1042 setContextMenuInfo(EntryContextMenuInfo info)1043 public void setContextMenuInfo(EntryContextMenuInfo info) { 1044 mEntryContextMenuInfo = info; 1045 } 1046 1047 @Override getContextMenuInfo()1048 protected ContextMenuInfo getContextMenuInfo() { 1049 return mEntryContextMenuInfo; 1050 } 1051 } 1052 1053 public static final class EntryContextMenuInfo implements ContextMenuInfo { 1054 private final String mCopyText; 1055 private final String mCopyLabel; 1056 private final String mMimeType; 1057 private final long mId; 1058 private final boolean mIsSuperPrimary; 1059 EntryContextMenuInfo(String copyText, String copyLabel, String mimeType, long id, boolean isSuperPrimary)1060 public EntryContextMenuInfo(String copyText, String copyLabel, String mimeType, long id, 1061 boolean isSuperPrimary) { 1062 mCopyText = copyText; 1063 mCopyLabel = copyLabel; 1064 mMimeType = mimeType; 1065 mId = id; 1066 mIsSuperPrimary = isSuperPrimary; 1067 } 1068 getCopyText()1069 public String getCopyText() { 1070 return mCopyText; 1071 } 1072 getCopyLabel()1073 public String getCopyLabel() { 1074 return mCopyLabel; 1075 } 1076 getMimeType()1077 public String getMimeType() { 1078 return mMimeType; 1079 } 1080 getId()1081 public long getId() { 1082 return mId; 1083 } 1084 isSuperPrimary()1085 public boolean isSuperPrimary() { 1086 return mIsSuperPrimary; 1087 } 1088 } 1089 1090 static final class EntryTag { 1091 private final int mId; 1092 private final Intent mIntent; 1093 EntryTag(int id, Intent intent)1094 public EntryTag(int id, Intent intent) { 1095 mId = id; 1096 mIntent = intent; 1097 } 1098 getId()1099 public int getId() { 1100 return mId; 1101 } 1102 getIntent()1103 public Intent getIntent() { 1104 return mIntent; 1105 } 1106 } 1107 1108 /** 1109 * This custom touch listener increases the touch area for the second and third icons, if 1110 * they are present. This is necessary to maintain other properties on an entry view, like 1111 * using a top padding on entry. Based off of {@link android.view.TouchDelegate} 1112 */ 1113 private static final class EntryTouchListener implements View.OnTouchListener { 1114 private final View mEntry; 1115 private final ImageView mAlternateIcon; 1116 private final ImageView mThirdIcon; 1117 /** mTouchedView locks in a view on touch down */ 1118 private View mTouchedView; 1119 /** mSlop adds some space to account for touches that are just outside the hit area */ 1120 private int mSlop; 1121 EntryTouchListener(View entry, ImageView alternateIcon, ImageView thirdIcon)1122 public EntryTouchListener(View entry, ImageView alternateIcon, ImageView thirdIcon) { 1123 mEntry = entry; 1124 mAlternateIcon = alternateIcon; 1125 mThirdIcon = thirdIcon; 1126 mSlop = ViewConfiguration.get(entry.getContext()).getScaledTouchSlop(); 1127 } 1128 1129 @Override onTouch(View v, MotionEvent event)1130 public boolean onTouch(View v, MotionEvent event) { 1131 View touchedView = mTouchedView; 1132 boolean sendToTouched = false; 1133 boolean hit = true; 1134 boolean handled = false; 1135 1136 switch (event.getAction()) { 1137 case MotionEvent.ACTION_DOWN: 1138 if (hitThirdIcon(event)) { 1139 mTouchedView = mThirdIcon; 1140 sendToTouched = true; 1141 } else if (hitAlternateIcon(event)) { 1142 mTouchedView = mAlternateIcon; 1143 sendToTouched = true; 1144 } else { 1145 mTouchedView = mEntry; 1146 sendToTouched = false; 1147 } 1148 touchedView = mTouchedView; 1149 break; 1150 case MotionEvent.ACTION_UP: 1151 case MotionEvent.ACTION_MOVE: 1152 sendToTouched = mTouchedView != null && mTouchedView != mEntry; 1153 if (sendToTouched) { 1154 final Rect slopBounds = new Rect(); 1155 touchedView.getHitRect(slopBounds); 1156 slopBounds.inset(-mSlop, -mSlop); 1157 if (!slopBounds.contains((int) event.getX(), (int) event.getY())) { 1158 hit = false; 1159 } 1160 } 1161 break; 1162 case MotionEvent.ACTION_CANCEL: 1163 sendToTouched = mTouchedView != null && mTouchedView != mEntry; 1164 mTouchedView = null; 1165 break; 1166 } 1167 if (sendToTouched) { 1168 if (hit) { 1169 event.setLocation(touchedView.getWidth() / 2, touchedView.getHeight() / 2); 1170 } else { 1171 // Offset event coordinates to be outside the target view (in case it does 1172 // something like tracking pressed state) 1173 event.setLocation(-(mSlop * 2), -(mSlop * 2)); 1174 } 1175 handled = touchedView.dispatchTouchEvent(event); 1176 } 1177 return handled; 1178 } 1179 hitThirdIcon(MotionEvent event)1180 private boolean hitThirdIcon(MotionEvent event) { 1181 if (mEntry.getLayoutDirection() == View.LAYOUT_DIRECTION_RTL) { 1182 return mThirdIcon.getVisibility() == View.VISIBLE && 1183 event.getX() < mThirdIcon.getRight(); 1184 } else { 1185 return mThirdIcon.getVisibility() == View.VISIBLE && 1186 event.getX() > mThirdIcon.getLeft(); 1187 } 1188 } 1189 1190 /** 1191 * Should be used after checking if third icon was hit 1192 */ hitAlternateIcon(MotionEvent event)1193 private boolean hitAlternateIcon(MotionEvent event) { 1194 // LayoutParams used to add the start margin to the touch area 1195 final RelativeLayout.LayoutParams alternateIconParams = 1196 (RelativeLayout.LayoutParams) mAlternateIcon.getLayoutParams(); 1197 if (mEntry.getLayoutDirection() == View.LAYOUT_DIRECTION_RTL) { 1198 return mAlternateIcon.getVisibility() == View.VISIBLE && 1199 event.getX() < mAlternateIcon.getRight() + alternateIconParams.rightMargin; 1200 } else { 1201 return mAlternateIcon.getVisibility() == View.VISIBLE && 1202 event.getX() > mAlternateIcon.getLeft() - alternateIconParams.leftMargin; 1203 } 1204 } 1205 } 1206 } 1207