1 /* 2 * Copyright (C) 2011 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 17 package com.android.contacts.detail; 18 19 import android.content.ContentUris; 20 import android.content.Context; 21 import android.content.pm.PackageManager; 22 import android.content.pm.PackageManager.NameNotFoundException; 23 import android.content.res.Resources; 24 import android.content.res.Resources.NotFoundException; 25 import android.graphics.drawable.Drawable; 26 import android.net.Uri; 27 import android.provider.ContactsContract; 28 import android.provider.ContactsContract.DisplayNameSources; 29 import android.provider.ContactsContract.StreamItems; 30 import android.text.Html; 31 import android.text.Html.ImageGetter; 32 import android.text.TextUtils; 33 import android.util.Log; 34 import android.view.LayoutInflater; 35 import android.view.MenuItem; 36 import android.view.View; 37 import android.view.ViewGroup; 38 import android.widget.ImageView; 39 import android.widget.ListView; 40 import android.widget.TextView; 41 42 import com.android.contacts.ContactPhotoManager; 43 import com.android.contacts.R; 44 import com.android.contacts.model.Contact; 45 import com.android.contacts.model.RawContact; 46 import com.android.contacts.model.dataitem.DataItem; 47 import com.android.contacts.model.dataitem.OrganizationDataItem; 48 import com.android.contacts.preference.ContactsPreferences; 49 import com.android.contacts.util.ContactBadgeUtil; 50 import com.android.contacts.util.HtmlUtils; 51 import com.android.contacts.util.MoreMath; 52 import com.android.contacts.util.StreamItemEntry; 53 import com.android.contacts.util.StreamItemPhotoEntry; 54 import com.google.common.annotations.VisibleForTesting; 55 import com.google.common.collect.Iterables; 56 57 import java.util.List; 58 59 /** 60 * This class contains utility methods to bind high-level contact details 61 * (meaning name, phonetic name, job, and attribution) from a 62 * {@link Contact} data object to appropriate {@link View}s. 63 */ 64 public class ContactDetailDisplayUtils { 65 private static final String TAG = "ContactDetailDisplayUtils"; 66 67 /** 68 * Tag object used for stream item photos. 69 */ 70 public static class StreamPhotoTag { 71 public final StreamItemEntry streamItem; 72 public final StreamItemPhotoEntry streamItemPhoto; 73 StreamPhotoTag(StreamItemEntry streamItem, StreamItemPhotoEntry streamItemPhoto)74 public StreamPhotoTag(StreamItemEntry streamItem, StreamItemPhotoEntry streamItemPhoto) { 75 this.streamItem = streamItem; 76 this.streamItemPhoto = streamItemPhoto; 77 } 78 getStreamItemPhotoUri()79 public Uri getStreamItemPhotoUri() { 80 final Uri.Builder builder = StreamItems.CONTENT_URI.buildUpon(); 81 ContentUris.appendId(builder, streamItem.getId()); 82 builder.appendPath(StreamItems.StreamItemPhotos.CONTENT_DIRECTORY); 83 ContentUris.appendId(builder, streamItemPhoto.getId()); 84 return builder.build(); 85 } 86 } 87 ContactDetailDisplayUtils()88 private ContactDetailDisplayUtils() { 89 // Disallow explicit creation of this class. 90 } 91 92 /** 93 * Returns the display name of the contact, using the current display order setting. 94 * Returns res/string/missing_name if there is no display name. 95 */ getDisplayName(Context context, Contact contactData)96 public static CharSequence getDisplayName(Context context, Contact contactData) { 97 CharSequence displayName = contactData.getDisplayName(); 98 CharSequence altDisplayName = contactData.getAltDisplayName(); 99 ContactsPreferences prefs = new ContactsPreferences(context); 100 CharSequence styledName = ""; 101 if (!TextUtils.isEmpty(displayName) && !TextUtils.isEmpty(altDisplayName)) { 102 if (prefs.getDisplayOrder() == ContactsContract.Preferences.DISPLAY_ORDER_PRIMARY) { 103 styledName = displayName; 104 } else { 105 styledName = altDisplayName; 106 } 107 } else { 108 styledName = context.getResources().getString(R.string.missing_name); 109 } 110 return styledName; 111 } 112 113 /** 114 * Returns the phonetic name of the contact or null if there isn't one. 115 */ getPhoneticName(Context context, Contact contactData)116 public static String getPhoneticName(Context context, Contact contactData) { 117 String phoneticName = contactData.getPhoneticName(); 118 if (!TextUtils.isEmpty(phoneticName)) { 119 return phoneticName; 120 } 121 return null; 122 } 123 124 /** 125 * Returns the attribution string for the contact, which may specify the contact directory that 126 * the contact came from. Returns null if there is none applicable. 127 */ getAttribution(Context context, Contact contactData)128 public static String getAttribution(Context context, Contact contactData) { 129 if (contactData.isDirectoryEntry()) { 130 String directoryDisplayName = contactData.getDirectoryDisplayName(); 131 String directoryType = contactData.getDirectoryType(); 132 String displayName = !TextUtils.isEmpty(directoryDisplayName) 133 ? directoryDisplayName 134 : directoryType; 135 return context.getString(R.string.contact_directory_description, displayName); 136 } 137 return null; 138 } 139 140 /** 141 * Returns the organization of the contact. If several organizations are given, 142 * the first one is used. Returns null if not applicable. 143 */ getCompany(Context context, Contact contactData)144 public static String getCompany(Context context, Contact contactData) { 145 final boolean displayNameIsOrganization = contactData.getDisplayNameSource() 146 == DisplayNameSources.ORGANIZATION; 147 for (RawContact rawContact : contactData.getRawContacts()) { 148 for (DataItem dataItem : Iterables.filter( 149 rawContact.getDataItems(), OrganizationDataItem.class)) { 150 OrganizationDataItem organization = (OrganizationDataItem) dataItem; 151 final String company = organization.getCompany(); 152 final String title = organization.getTitle(); 153 final String combined; 154 // We need to show company and title in a combined string. However, if the 155 // DisplayName is already the organization, it mirrors company or (if company 156 // is empty title). Make sure we don't show what's already shown as DisplayName 157 if (TextUtils.isEmpty(company)) { 158 combined = displayNameIsOrganization ? null : title; 159 } else { 160 if (TextUtils.isEmpty(title)) { 161 combined = displayNameIsOrganization ? null : company; 162 } else { 163 if (displayNameIsOrganization) { 164 combined = title; 165 } else { 166 combined = context.getString( 167 R.string.organization_company_and_title, 168 company, title); 169 } 170 } 171 } 172 173 if (!TextUtils.isEmpty(combined)) { 174 return combined; 175 } 176 } 177 } 178 return null; 179 } 180 181 /** 182 * Sets the starred state of this contact. 183 */ configureStarredImageView(ImageView starredView, boolean isDirectoryEntry, boolean isUserProfile, boolean isStarred)184 public static void configureStarredImageView(ImageView starredView, boolean isDirectoryEntry, 185 boolean isUserProfile, boolean isStarred) { 186 // Check if the starred state should be visible 187 if (!isDirectoryEntry && !isUserProfile) { 188 starredView.setVisibility(View.VISIBLE); 189 final int resId = isStarred 190 ? R.drawable.btn_star_on_normal_holo_light 191 : R.drawable.btn_star_off_normal_holo_light; 192 starredView.setImageResource(resId); 193 starredView.setTag(isStarred); 194 starredView.setContentDescription(starredView.getResources().getString( 195 isStarred ? R.string.menu_removeStar : R.string.menu_addStar)); 196 } else { 197 starredView.setVisibility(View.GONE); 198 } 199 } 200 201 /** 202 * Sets the starred state of this contact. 203 */ configureStarredMenuItem(MenuItem starredMenuItem, boolean isDirectoryEntry, boolean isUserProfile, boolean isStarred)204 public static void configureStarredMenuItem(MenuItem starredMenuItem, boolean isDirectoryEntry, 205 boolean isUserProfile, boolean isStarred) { 206 // Check if the starred state should be visible 207 if (!isDirectoryEntry && !isUserProfile) { 208 starredMenuItem.setVisible(true); 209 final int resId = isStarred 210 ? R.drawable.btn_star_on_normal_holo_dark 211 : R.drawable.btn_star_off_normal_holo_dark; 212 starredMenuItem.setIcon(resId); 213 starredMenuItem.setChecked(isStarred); 214 starredMenuItem.setTitle(isStarred ? R.string.menu_removeStar : R.string.menu_addStar); 215 } else { 216 starredMenuItem.setVisible(false); 217 } 218 } 219 220 /** 221 * Set the social snippet text. If there isn't one, then set the view to gone. 222 */ setSocialSnippet(Context context, Contact contactData, TextView statusView, ImageView statusPhotoView)223 public static void setSocialSnippet(Context context, Contact contactData, TextView statusView, 224 ImageView statusPhotoView) { 225 if (statusView == null) { 226 return; 227 } 228 229 CharSequence snippet = null; 230 String photoUri = null; 231 if (!contactData.getStreamItems().isEmpty()) { 232 StreamItemEntry firstEntry = contactData.getStreamItems().get(0); 233 snippet = HtmlUtils.fromHtml(context, firstEntry.getText()); 234 if (!firstEntry.getPhotos().isEmpty()) { 235 StreamItemPhotoEntry firstPhoto = firstEntry.getPhotos().get(0); 236 photoUri = firstPhoto.getPhotoUri(); 237 238 // If displaying an image, hide the snippet text. 239 snippet = null; 240 } 241 } 242 setDataOrHideIfNone(snippet, statusView); 243 if (photoUri != null) { 244 ContactPhotoManager.getInstance(context).loadPhoto( 245 statusPhotoView, Uri.parse(photoUri), -1, false, 246 ContactPhotoManager.DEFAULT_BLANK); 247 statusPhotoView.setVisibility(View.VISIBLE); 248 } else { 249 statusPhotoView.setVisibility(View.GONE); 250 } 251 } 252 253 /** Creates the view that represents a stream item. */ createStreamItemView(LayoutInflater inflater, Context context, View convertView, StreamItemEntry streamItem, View.OnClickListener photoClickListener)254 public static View createStreamItemView(LayoutInflater inflater, Context context, 255 View convertView, StreamItemEntry streamItem, View.OnClickListener photoClickListener) { 256 257 // Try to recycle existing views. 258 final View container; 259 if (convertView != null) { 260 container = convertView; 261 } else { 262 container = inflater.inflate(R.layout.stream_item_container, null, false); 263 } 264 265 final ContactPhotoManager contactPhotoManager = ContactPhotoManager.getInstance(context); 266 final List<StreamItemPhotoEntry> photos = streamItem.getPhotos(); 267 final int photoCount = photos.size(); 268 269 // Add the text part. 270 addStreamItemText(context, streamItem, container); 271 272 // Add images. 273 final ViewGroup imageRows = (ViewGroup) container.findViewById(R.id.stream_item_image_rows); 274 275 if (photoCount == 0) { 276 // This stream item only has text. 277 imageRows.setVisibility(View.GONE); 278 } else { 279 // This stream item has text and photos. 280 imageRows.setVisibility(View.VISIBLE); 281 282 // Number of image rows needed, which is cailing(photoCount / 2) 283 final int numImageRows = (photoCount + 1) / 2; 284 285 // Actual image rows. 286 final int numOldImageRows = imageRows.getChildCount(); 287 288 // Make sure we have enough stream_item_row_images. 289 if (numOldImageRows == numImageRows) { 290 // Great, we have the just enough number of rows... 291 292 } else if (numOldImageRows < numImageRows) { 293 // Need to add more image rows. 294 for (int i = numOldImageRows; i < numImageRows; i++) { 295 View imageRow = inflater.inflate(R.layout.stream_item_row_images, imageRows, 296 true); 297 } 298 } else { 299 // We have exceeding image rows. Hide them. 300 for (int i = numImageRows; i < numOldImageRows; i++) { 301 imageRows.getChildAt(i).setVisibility(View.GONE); 302 } 303 } 304 305 // Put images, two by two. 306 for (int i = 0; i < photoCount; i += 2) { 307 final View imageRow = imageRows.getChildAt(i / 2); 308 // Reused image rows may not visible, so make sure they're shown. 309 imageRow.setVisibility(View.VISIBLE); 310 311 // Show first image. 312 loadPhoto(contactPhotoManager, streamItem, photos.get(i), imageRow, 313 R.id.stream_item_first_image, photoClickListener); 314 final View secondContainer = imageRow.findViewById(R.id.second_image_container); 315 if (i + 1 < photoCount) { 316 // Show the second image too. 317 loadPhoto(contactPhotoManager, streamItem, photos.get(i + 1), imageRow, 318 R.id.stream_item_second_image, photoClickListener); 319 secondContainer.setVisibility(View.VISIBLE); 320 } else { 321 // Hide the second image, but it still has to occupy the space. 322 secondContainer.setVisibility(View.INVISIBLE); 323 } 324 } 325 } 326 327 return container; 328 } 329 330 /** Loads a photo into an image view. The image view is identified by the given id. */ loadPhoto(ContactPhotoManager contactPhotoManager, final StreamItemEntry streamItem, final StreamItemPhotoEntry streamItemPhoto, View photoContainer, int imageViewId, View.OnClickListener photoClickListener)331 private static void loadPhoto(ContactPhotoManager contactPhotoManager, 332 final StreamItemEntry streamItem, final StreamItemPhotoEntry streamItemPhoto, 333 View photoContainer, int imageViewId, View.OnClickListener photoClickListener) { 334 final View frame = photoContainer.findViewById(imageViewId); 335 final View pushLayerView = frame.findViewById(R.id.push_layer); 336 final ImageView imageView = (ImageView) frame.findViewById(R.id.image); 337 if (photoClickListener != null) { 338 pushLayerView.setOnClickListener(photoClickListener); 339 pushLayerView.setTag(new StreamPhotoTag(streamItem, streamItemPhoto)); 340 pushLayerView.setFocusable(true); 341 pushLayerView.setEnabled(true); 342 } else { 343 pushLayerView.setOnClickListener(null); 344 pushLayerView.setTag(null); 345 pushLayerView.setFocusable(false); 346 // setOnClickListener makes it clickable, so we need to overwrite it 347 pushLayerView.setClickable(false); 348 pushLayerView.setEnabled(false); 349 } 350 contactPhotoManager.loadPhoto(imageView, Uri.parse(streamItemPhoto.getPhotoUri()), -1, 351 false, ContactPhotoManager.DEFAULT_BLANK); 352 } 353 354 @VisibleForTesting addStreamItemText(Context context, StreamItemEntry streamItem, View rootView)355 static View addStreamItemText(Context context, StreamItemEntry streamItem, View rootView) { 356 TextView htmlView = (TextView) rootView.findViewById(R.id.stream_item_html); 357 TextView attributionView = (TextView) rootView.findViewById( 358 R.id.stream_item_attribution); 359 TextView commentsView = (TextView) rootView.findViewById(R.id.stream_item_comments); 360 ImageGetter imageGetter = new DefaultImageGetter(context.getPackageManager()); 361 362 // Stream item text 363 setDataOrHideIfNone(streamItem.getDecodedText(), htmlView); 364 // Attribution 365 setDataOrHideIfNone(ContactBadgeUtil.getSocialDate(streamItem, context), 366 attributionView); 367 // Comments 368 setDataOrHideIfNone(streamItem.getDecodedComments(), commentsView); 369 return rootView; 370 } 371 372 /** 373 * Sets the display name of this contact to the given {@link TextView}. If 374 * there is none, then set the view to gone. 375 */ setDisplayName(Context context, Contact contactData, TextView textView)376 public static void setDisplayName(Context context, Contact contactData, TextView textView) { 377 if (textView == null) { 378 return; 379 } 380 setDataOrHideIfNone(getDisplayName(context, contactData), textView); 381 } 382 383 /** 384 * Sets the company and job title of this contact to the given {@link TextView}. If 385 * there is none, then set the view to gone. 386 */ setCompanyName(Context context, Contact contactData, TextView textView)387 public static void setCompanyName(Context context, Contact contactData, TextView textView) { 388 if (textView == null) { 389 return; 390 } 391 setDataOrHideIfNone(getCompany(context, contactData), textView); 392 } 393 394 /** 395 * Sets the phonetic name of this contact to the given {@link TextView}. If 396 * there is none, then set the view to gone. 397 */ setPhoneticName(Context context, Contact contactData, TextView textView)398 public static void setPhoneticName(Context context, Contact contactData, TextView textView) { 399 if (textView == null) { 400 return; 401 } 402 setDataOrHideIfNone(getPhoneticName(context, contactData), textView); 403 } 404 405 /** 406 * Sets the attribution contact to the given {@link TextView}. If 407 * there is none, then set the view to gone. 408 */ setAttribution(Context context, Contact contactData, TextView textView)409 public static void setAttribution(Context context, Contact contactData, TextView textView) { 410 if (textView == null) { 411 return; 412 } 413 setDataOrHideIfNone(getAttribution(context, contactData), textView); 414 } 415 416 /** 417 * Helper function to display the given text in the {@link TextView} or 418 * hides the {@link TextView} if the text is empty or null. 419 */ setDataOrHideIfNone(CharSequence textToDisplay, TextView textView)420 private static void setDataOrHideIfNone(CharSequence textToDisplay, TextView textView) { 421 if (!TextUtils.isEmpty(textToDisplay)) { 422 textView.setText(textToDisplay); 423 textView.setVisibility(View.VISIBLE); 424 } else { 425 textView.setText(null); 426 textView.setVisibility(View.GONE); 427 } 428 } 429 430 private static Html.ImageGetter sImageGetter; 431 getImageGetter(Context context)432 public static Html.ImageGetter getImageGetter(Context context) { 433 if (sImageGetter == null) { 434 sImageGetter = new DefaultImageGetter(context.getPackageManager()); 435 } 436 return sImageGetter; 437 } 438 439 /** Fetcher for images from resources to be included in HTML text. */ 440 private static class DefaultImageGetter implements Html.ImageGetter { 441 /** The scheme used to load resources. */ 442 private static final String RES_SCHEME = "res"; 443 444 private final PackageManager mPackageManager; 445 DefaultImageGetter(PackageManager packageManager)446 public DefaultImageGetter(PackageManager packageManager) { 447 mPackageManager = packageManager; 448 } 449 450 @Override getDrawable(String source)451 public Drawable getDrawable(String source) { 452 // Returning null means that a default image will be used. 453 Uri uri; 454 try { 455 uri = Uri.parse(source); 456 } catch (Throwable e) { 457 Log.d(TAG, "Could not parse image source: " + source); 458 return null; 459 } 460 if (!RES_SCHEME.equals(uri.getScheme())) { 461 Log.d(TAG, "Image source does not correspond to a resource: " + source); 462 return null; 463 } 464 // The URI authority represents the package name. 465 String packageName = uri.getAuthority(); 466 467 Resources resources = getResourcesForResourceName(packageName); 468 if (resources == null) { 469 Log.d(TAG, "Could not parse image source: " + source); 470 return null; 471 } 472 473 List<String> pathSegments = uri.getPathSegments(); 474 if (pathSegments.size() != 1) { 475 Log.d(TAG, "Could not parse image source: " + source); 476 return null; 477 } 478 479 final String name = pathSegments.get(0); 480 final int resId = resources.getIdentifier(name, "drawable", packageName); 481 482 if (resId == 0) { 483 // Use the default image icon in this case. 484 Log.d(TAG, "Cannot resolve resource identifier: " + source); 485 return null; 486 } 487 488 try { 489 return getResourceDrawable(resources, resId); 490 } catch (NotFoundException e) { 491 Log.d(TAG, "Resource not found: " + source, e); 492 return null; 493 } 494 } 495 496 /** Returns the drawable associated with the given id. */ getResourceDrawable(Resources resources, int resId)497 private Drawable getResourceDrawable(Resources resources, int resId) 498 throws NotFoundException { 499 Drawable drawable = resources.getDrawable(resId); 500 drawable.setBounds(0, 0, drawable.getIntrinsicWidth(), drawable.getIntrinsicHeight()); 501 return drawable; 502 } 503 504 /** Returns the {@link Resources} of the package of the given resource name. */ getResourcesForResourceName(String packageName)505 private Resources getResourcesForResourceName(String packageName) { 506 try { 507 return mPackageManager.getResourcesForApplication(packageName); 508 } catch (NameNotFoundException e) { 509 Log.d(TAG, "Could not find package: " + packageName); 510 return null; 511 } 512 } 513 } 514 515 /** 516 * Sets an alpha value on the view. 517 */ setAlphaOnViewBackground(View view, float alpha)518 public static void setAlphaOnViewBackground(View view, float alpha) { 519 if (view != null) { 520 // Convert alpha layer to a black background HEX color with an alpha value for better 521 // performance (i.e. use setBackgroundColor() instead of setAlpha()) 522 view.setBackgroundColor((int) (MoreMath.clamp(alpha, 0.0f, 1.0f) * 255) << 24); 523 } 524 } 525 526 /** 527 * Returns the top coordinate of the first item in the {@link ListView}. If the first item 528 * in the {@link ListView} is not visible or there are no children in the list, then return 529 * Integer.MIN_VALUE. Note that the returned value will be <= 0 because the first item in the 530 * list cannot have a positive offset. 531 */ getFirstListItemOffset(ListView listView)532 public static int getFirstListItemOffset(ListView listView) { 533 if (listView == null || listView.getChildCount() == 0 || 534 listView.getFirstVisiblePosition() != 0) { 535 return Integer.MIN_VALUE; 536 } 537 return listView.getChildAt(0).getTop(); 538 } 539 540 /** 541 * Tries to scroll the first item in the list to the given offset (this can be a no-op if the 542 * list is already in the correct position). 543 * @param listView that should be scrolled 544 * @param offset which should be <= 0 545 */ requestToMoveToOffset(ListView listView, int offset)546 public static void requestToMoveToOffset(ListView listView, int offset) { 547 // We try to offset the list if the first item in the list is showing (which is presumed 548 // to have a larger height than the desired offset). If the first item in the list is not 549 // visible, then we simply do not scroll the list at all (since it can get complicated to 550 // compute how many items in the list will equal the given offset). Potentially 551 // some animation elsewhere will make the transition smoother for the user to compensate 552 // for this simplification. 553 if (listView == null || listView.getChildCount() == 0 || 554 listView.getFirstVisiblePosition() != 0 || offset > 0) { 555 return; 556 } 557 558 // As an optimization, check if the first item is already at the given offset. 559 if (listView.getChildAt(0).getTop() == offset) { 560 return; 561 } 562 563 listView.setSelectionFromTop(0, offset); 564 } 565 } 566