• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2010 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.list;
17 
18 import android.content.Context;
19 import android.content.CursorLoader;
20 import android.database.Cursor;
21 import android.net.Uri;
22 import android.os.Bundle;
23 import android.provider.ContactsContract;
24 import android.provider.ContactsContract.ContactCounts;
25 import android.provider.ContactsContract.Contacts;
26 import android.provider.ContactsContract.Directory;
27 import android.text.TextUtils;
28 import android.util.Log;
29 import android.view.LayoutInflater;
30 import android.view.View;
31 import android.view.ViewGroup;
32 import android.widget.QuickContactBadge;
33 import android.widget.SectionIndexer;
34 import android.widget.TextView;
35 
36 import com.android.contacts.ContactPhotoManager;
37 import com.android.contacts.R;
38 import com.android.contacts.widget.IndexerListAdapter;
39 
40 import java.util.HashSet;
41 
42 /**
43  * Common base class for various contact-related lists, e.g. contact list, phone number list
44  * etc.
45  */
46 public abstract class ContactEntryListAdapter extends IndexerListAdapter {
47 
48     private static final String TAG = "ContactEntryListAdapter";
49 
50     /**
51      * Indicates whether the {@link Directory#LOCAL_INVISIBLE} directory should
52      * be included in the search.
53      */
54     private static final boolean LOCAL_INVISIBLE_DIRECTORY_ENABLED = false;
55 
56     private int mDisplayOrder;
57     private int mSortOrder;
58 
59     private boolean mDisplayPhotos;
60     private boolean mQuickContactEnabled;
61 
62     /**
63      * indicates if contact queries include profile
64      */
65     private boolean mIncludeProfile;
66 
67     /**
68      * indicates if query results includes a profile
69      */
70     private boolean mProfileExists;
71 
72     private ContactPhotoManager mPhotoLoader;
73 
74     private String mQueryString;
75     private char[] mUpperCaseQueryString;
76     private boolean mSearchMode;
77     private int mDirectorySearchMode;
78     private int mDirectoryResultLimit = Integer.MAX_VALUE;
79 
80     private boolean mLoading = true;
81     private boolean mEmptyListEnabled = true;
82 
83     private boolean mSelectionVisible;
84 
85     private ContactListFilter mFilter;
86     private String mContactsCount = "";
87     private boolean mDarkTheme = false;
88 
89     /** Resource used to provide header-text for default filter. */
90     private CharSequence mDefaultFilterHeaderText;
91 
ContactEntryListAdapter(Context context)92     public ContactEntryListAdapter(Context context) {
93         super(context);
94         addPartitions();
95         setDefaultFilterHeaderText(R.string.local_search_label);
96     }
97 
setDefaultFilterHeaderText(int resourceId)98     protected void setDefaultFilterHeaderText(int resourceId) {
99         mDefaultFilterHeaderText = getContext().getResources().getText(resourceId);
100     }
101 
102     @Override
createPinnedSectionHeaderView(Context context, ViewGroup parent)103     protected View createPinnedSectionHeaderView(Context context, ViewGroup parent) {
104         return new ContactListPinnedHeaderView(context, null);
105     }
106 
107     @Override
setPinnedSectionTitle(View pinnedHeaderView, String title)108     protected void setPinnedSectionTitle(View pinnedHeaderView, String title) {
109         ((ContactListPinnedHeaderView)pinnedHeaderView).setSectionHeader(title);
110     }
111 
112     @Override
setPinnedHeaderContactsCount(View header)113     protected void setPinnedHeaderContactsCount(View header) {
114         // Update the header with the contacts count only if a profile header exists
115         // otherwise, the contacts count are shown in the empty profile header view
116         if (mProfileExists) {
117             ((ContactListPinnedHeaderView)header).setCountView(mContactsCount);
118         } else {
119             clearPinnedHeaderContactsCount(header);
120         }
121     }
122 
123     @Override
clearPinnedHeaderContactsCount(View header)124     protected void clearPinnedHeaderContactsCount(View header) {
125         ((ContactListPinnedHeaderView)header).setCountView(null);
126     }
127 
addPartitions()128     protected void addPartitions() {
129         addPartition(createDefaultDirectoryPartition());
130     }
131 
createDefaultDirectoryPartition()132     protected DirectoryPartition createDefaultDirectoryPartition() {
133         DirectoryPartition partition = new DirectoryPartition(true, true);
134         partition.setDirectoryId(Directory.DEFAULT);
135         partition.setDirectoryType(getContext().getString(R.string.contactsList));
136         partition.setPriorityDirectory(true);
137         partition.setPhotoSupported(true);
138         return partition;
139     }
140 
141     /**
142      * Remove all directories after the default directory. This is typically used when contacts
143      * list screens are asked to exit the search mode and thus need to remove all remote directory
144      * results for the search.
145      *
146      * This code assumes that the default directory and directories before that should not be
147      * deleted (e.g. Join screen has "suggested contacts" directory before the default director,
148      * and we should not remove the directory).
149      */
removeDirectoriesAfterDefault()150     /* package */ void removeDirectoriesAfterDefault() {
151         final int partitionCount = getPartitionCount();
152         for (int i = partitionCount - 1; i >= 0; i--) {
153             final Partition partition = getPartition(i);
154             if ((partition instanceof DirectoryPartition)
155                     && ((DirectoryPartition) partition).getDirectoryId() == Directory.DEFAULT) {
156                 break;
157             } else {
158                 removePartition(i);
159             }
160         }
161     }
162 
getPartitionByDirectoryId(long id)163     private int getPartitionByDirectoryId(long id) {
164         int count = getPartitionCount();
165         for (int i = 0; i < count; i++) {
166             Partition partition = getPartition(i);
167             if (partition instanceof DirectoryPartition) {
168                 if (((DirectoryPartition)partition).getDirectoryId() == id) {
169                     return i;
170                 }
171             }
172         }
173         return -1;
174     }
175 
getContactDisplayName(int position)176     public abstract String getContactDisplayName(int position);
configureLoader(CursorLoader loader, long directoryId)177     public abstract void configureLoader(CursorLoader loader, long directoryId);
178 
179     /**
180      * Marks all partitions as "loading"
181      */
onDataReload()182     public void onDataReload() {
183         boolean notify = false;
184         int count = getPartitionCount();
185         for (int i = 0; i < count; i++) {
186             Partition partition = getPartition(i);
187             if (partition instanceof DirectoryPartition) {
188                 DirectoryPartition directoryPartition = (DirectoryPartition)partition;
189                 if (!directoryPartition.isLoading()) {
190                     notify = true;
191                 }
192                 directoryPartition.setStatus(DirectoryPartition.STATUS_NOT_LOADED);
193             }
194         }
195         if (notify) {
196             notifyDataSetChanged();
197         }
198     }
199 
200     @Override
clearPartitions()201     public void clearPartitions() {
202         int count = getPartitionCount();
203         for (int i = 0; i < count; i++) {
204             Partition partition = getPartition(i);
205             if (partition instanceof DirectoryPartition) {
206                 DirectoryPartition directoryPartition = (DirectoryPartition)partition;
207                 directoryPartition.setStatus(DirectoryPartition.STATUS_NOT_LOADED);
208             }
209         }
210         super.clearPartitions();
211     }
212 
isSearchMode()213     public boolean isSearchMode() {
214         return mSearchMode;
215     }
216 
setSearchMode(boolean flag)217     public void setSearchMode(boolean flag) {
218         mSearchMode = flag;
219     }
220 
getQueryString()221     public String getQueryString() {
222         return mQueryString;
223     }
224 
setQueryString(String queryString)225     public void setQueryString(String queryString) {
226         mQueryString = queryString;
227         if (TextUtils.isEmpty(queryString)) {
228             mUpperCaseQueryString = null;
229         } else {
230             mUpperCaseQueryString = queryString.toUpperCase().toCharArray();
231         }
232     }
233 
getUpperCaseQueryString()234     public char[] getUpperCaseQueryString() {
235         return mUpperCaseQueryString;
236     }
237 
getDirectorySearchMode()238     public int getDirectorySearchMode() {
239         return mDirectorySearchMode;
240     }
241 
setDirectorySearchMode(int mode)242     public void setDirectorySearchMode(int mode) {
243         mDirectorySearchMode = mode;
244     }
245 
getDirectoryResultLimit()246     public int getDirectoryResultLimit() {
247         return mDirectoryResultLimit;
248     }
249 
setDirectoryResultLimit(int limit)250     public void setDirectoryResultLimit(int limit) {
251         this.mDirectoryResultLimit = limit;
252     }
253 
getContactNameDisplayOrder()254     public int getContactNameDisplayOrder() {
255         return mDisplayOrder;
256     }
257 
setContactNameDisplayOrder(int displayOrder)258     public void setContactNameDisplayOrder(int displayOrder) {
259         mDisplayOrder = displayOrder;
260     }
261 
getSortOrder()262     public int getSortOrder() {
263         return mSortOrder;
264     }
265 
setSortOrder(int sortOrder)266     public void setSortOrder(int sortOrder) {
267         mSortOrder = sortOrder;
268     }
269 
setPhotoLoader(ContactPhotoManager photoLoader)270     public void setPhotoLoader(ContactPhotoManager photoLoader) {
271         mPhotoLoader = photoLoader;
272     }
273 
getPhotoLoader()274     protected ContactPhotoManager getPhotoLoader() {
275         return mPhotoLoader;
276     }
277 
getDisplayPhotos()278     public boolean getDisplayPhotos() {
279         return mDisplayPhotos;
280     }
281 
setDisplayPhotos(boolean displayPhotos)282     public void setDisplayPhotos(boolean displayPhotos) {
283         mDisplayPhotos = displayPhotos;
284     }
285 
isEmptyListEnabled()286     public boolean isEmptyListEnabled() {
287         return mEmptyListEnabled;
288     }
289 
setEmptyListEnabled(boolean flag)290     public void setEmptyListEnabled(boolean flag) {
291         mEmptyListEnabled = flag;
292     }
293 
isSelectionVisible()294     public boolean isSelectionVisible() {
295         return mSelectionVisible;
296     }
297 
setSelectionVisible(boolean flag)298     public void setSelectionVisible(boolean flag) {
299         this.mSelectionVisible = flag;
300     }
301 
isQuickContactEnabled()302     public boolean isQuickContactEnabled() {
303         return mQuickContactEnabled;
304     }
305 
setQuickContactEnabled(boolean quickContactEnabled)306     public void setQuickContactEnabled(boolean quickContactEnabled) {
307         mQuickContactEnabled = quickContactEnabled;
308     }
309 
shouldIncludeProfile()310     public boolean shouldIncludeProfile() {
311         return mIncludeProfile;
312     }
313 
setIncludeProfile(boolean includeProfile)314     public void setIncludeProfile(boolean includeProfile) {
315         mIncludeProfile = includeProfile;
316     }
317 
setProfileExists(boolean exists)318     public void setProfileExists(boolean exists) {
319         mProfileExists = exists;
320         // Stick the "ME" header for the profile
321         if (exists) {
322             SectionIndexer indexer = getIndexer();
323             if (indexer != null) {
324                 ((ContactsSectionIndexer) indexer).setProfileHeader(
325                         getContext().getString(R.string.user_profile_contacts_list_header));
326             }
327         }
328     }
329 
hasProfile()330     public boolean hasProfile() {
331         return mProfileExists;
332     }
333 
setDarkTheme(boolean value)334     public void setDarkTheme(boolean value) {
335         mDarkTheme = value;
336     }
337 
configureDirectoryLoader(DirectoryListLoader loader)338     public void configureDirectoryLoader(DirectoryListLoader loader) {
339         loader.setDirectorySearchMode(mDirectorySearchMode);
340         loader.setLocalInvisibleDirectoryEnabled(LOCAL_INVISIBLE_DIRECTORY_ENABLED);
341     }
342 
343     /**
344      * Updates partitions according to the directory meta-data contained in the supplied
345      * cursor.
346      */
changeDirectories(Cursor cursor)347     public void changeDirectories(Cursor cursor) {
348         if (cursor.getCount() == 0) {
349             // Directory table must have at least local directory, without which this adapter will
350             // enter very weird state.
351             Log.e(TAG, "Directory search loader returned an empty cursor, which implies we have " +
352                     "no directory entries.", new RuntimeException());
353             return;
354         }
355         HashSet<Long> directoryIds = new HashSet<Long>();
356 
357         int idColumnIndex = cursor.getColumnIndex(Directory._ID);
358         int directoryTypeColumnIndex = cursor.getColumnIndex(DirectoryListLoader.DIRECTORY_TYPE);
359         int displayNameColumnIndex = cursor.getColumnIndex(Directory.DISPLAY_NAME);
360         int photoSupportColumnIndex = cursor.getColumnIndex(Directory.PHOTO_SUPPORT);
361 
362         // TODO preserve the order of partition to match those of the cursor
363         // Phase I: add new directories
364         cursor.moveToPosition(-1);
365         while (cursor.moveToNext()) {
366             long id = cursor.getLong(idColumnIndex);
367             directoryIds.add(id);
368             if (getPartitionByDirectoryId(id) == -1) {
369                 DirectoryPartition partition = new DirectoryPartition(false, true);
370                 partition.setDirectoryId(id);
371                 partition.setDirectoryType(cursor.getString(directoryTypeColumnIndex));
372                 partition.setDisplayName(cursor.getString(displayNameColumnIndex));
373                 int photoSupport = cursor.getInt(photoSupportColumnIndex);
374                 partition.setPhotoSupported(photoSupport == Directory.PHOTO_SUPPORT_THUMBNAIL_ONLY
375                         || photoSupport == Directory.PHOTO_SUPPORT_FULL);
376                 addPartition(partition);
377             }
378         }
379 
380         // Phase II: remove deleted directories
381         int count = getPartitionCount();
382         for (int i = count; --i >= 0; ) {
383             Partition partition = getPartition(i);
384             if (partition instanceof DirectoryPartition) {
385                 long id = ((DirectoryPartition)partition).getDirectoryId();
386                 if (!directoryIds.contains(id)) {
387                     removePartition(i);
388                 }
389             }
390         }
391 
392         invalidate();
393         notifyDataSetChanged();
394     }
395 
396     @Override
changeCursor(int partitionIndex, Cursor cursor)397     public void changeCursor(int partitionIndex, Cursor cursor) {
398         if (partitionIndex >= getPartitionCount()) {
399             // There is no partition for this data
400             return;
401         }
402 
403         Partition partition = getPartition(partitionIndex);
404         if (partition instanceof DirectoryPartition) {
405             ((DirectoryPartition)partition).setStatus(DirectoryPartition.STATUS_LOADED);
406         }
407 
408         if (mDisplayPhotos && mPhotoLoader != null && isPhotoSupported(partitionIndex)) {
409             mPhotoLoader.refreshCache();
410         }
411 
412         super.changeCursor(partitionIndex, cursor);
413 
414         if (isSectionHeaderDisplayEnabled() && partitionIndex == getIndexedPartition()) {
415             updateIndexer(cursor);
416         }
417     }
418 
changeCursor(Cursor cursor)419     public void changeCursor(Cursor cursor) {
420         changeCursor(0, cursor);
421     }
422 
423     /**
424      * Updates the indexer, which is used to produce section headers.
425      */
updateIndexer(Cursor cursor)426     private void updateIndexer(Cursor cursor) {
427         if (cursor == null) {
428             setIndexer(null);
429             return;
430         }
431 
432         Bundle bundle = cursor.getExtras();
433         if (bundle.containsKey(ContactCounts.EXTRA_ADDRESS_BOOK_INDEX_TITLES)) {
434             String sections[] =
435                     bundle.getStringArray(ContactCounts.EXTRA_ADDRESS_BOOK_INDEX_TITLES);
436             int counts[] = bundle.getIntArray(ContactCounts.EXTRA_ADDRESS_BOOK_INDEX_COUNTS);
437             setIndexer(new ContactsSectionIndexer(sections, counts));
438         } else {
439             setIndexer(null);
440         }
441     }
442 
443     @Override
getViewTypeCount()444     public int getViewTypeCount() {
445         // We need a separate view type for each item type, plus another one for
446         // each type with header, plus one for "other".
447         return getItemViewTypeCount() * 2 + 1;
448     }
449 
450     @Override
getItemViewType(int partitionIndex, int position)451     public int getItemViewType(int partitionIndex, int position) {
452         int type = super.getItemViewType(partitionIndex, position);
453         if (!isUserProfile(position)
454                 && isSectionHeaderDisplayEnabled()
455                 && partitionIndex == getIndexedPartition()) {
456             Placement placement = getItemPlacementInSection(position);
457             return placement.firstInSection ? type : getItemViewTypeCount() + type;
458         } else {
459             return type;
460         }
461     }
462 
463     @Override
isEmpty()464     public boolean isEmpty() {
465         // TODO
466 //        if (contactsListActivity.mProviderStatus != ProviderStatus.STATUS_NORMAL) {
467 //            return true;
468 //        }
469 
470         if (!mEmptyListEnabled) {
471             return false;
472         } else if (isSearchMode()) {
473             return TextUtils.isEmpty(getQueryString());
474         } else if (mLoading) {
475             // We don't want the empty state to show when loading.
476             return false;
477         } else {
478             return super.isEmpty();
479         }
480     }
481 
isLoading()482     public boolean isLoading() {
483         int count = getPartitionCount();
484         for (int i = 0; i < count; i++) {
485             Partition partition = getPartition(i);
486             if (partition instanceof DirectoryPartition
487                     && ((DirectoryPartition) partition).isLoading()) {
488                 return true;
489             }
490         }
491         return false;
492     }
493 
areAllPartitionsEmpty()494     public boolean areAllPartitionsEmpty() {
495         int count = getPartitionCount();
496         for (int i = 0; i < count; i++) {
497             if (!isPartitionEmpty(i)) {
498                 return false;
499             }
500         }
501         return true;
502     }
503 
504     /**
505      * Changes visibility parameters for the default directory partition.
506      */
configureDefaultPartition(boolean showIfEmpty, boolean hasHeader)507     public void configureDefaultPartition(boolean showIfEmpty, boolean hasHeader) {
508         int defaultPartitionIndex = -1;
509         int count = getPartitionCount();
510         for (int i = 0; i < count; i++) {
511             Partition partition = getPartition(i);
512             if (partition instanceof DirectoryPartition &&
513                     ((DirectoryPartition)partition).getDirectoryId() == Directory.DEFAULT) {
514                 defaultPartitionIndex = i;
515                 break;
516             }
517         }
518         if (defaultPartitionIndex != -1) {
519             setShowIfEmpty(defaultPartitionIndex, showIfEmpty);
520             setHasHeader(defaultPartitionIndex, hasHeader);
521         }
522     }
523 
524     @Override
newHeaderView(Context context, int partition, Cursor cursor, ViewGroup parent)525     protected View newHeaderView(Context context, int partition, Cursor cursor,
526             ViewGroup parent) {
527         LayoutInflater inflater = LayoutInflater.from(context);
528         return inflater.inflate(R.layout.directory_header, parent, false);
529     }
530 
531     @Override
bindHeaderView(View view, int partitionIndex, Cursor cursor)532     protected void bindHeaderView(View view, int partitionIndex, Cursor cursor) {
533         Partition partition = getPartition(partitionIndex);
534         if (!(partition instanceof DirectoryPartition)) {
535             return;
536         }
537 
538         DirectoryPartition directoryPartition = (DirectoryPartition)partition;
539         long directoryId = directoryPartition.getDirectoryId();
540         TextView labelTextView = (TextView)view.findViewById(R.id.label);
541         TextView displayNameTextView = (TextView)view.findViewById(R.id.display_name);
542         if (directoryId == Directory.DEFAULT || directoryId == Directory.LOCAL_INVISIBLE) {
543             labelTextView.setText(mDefaultFilterHeaderText);
544             displayNameTextView.setText(null);
545         } else {
546             labelTextView.setText(R.string.directory_search_label);
547             String directoryName = directoryPartition.getDisplayName();
548             String displayName = !TextUtils.isEmpty(directoryName)
549                     ? directoryName
550                     : directoryPartition.getDirectoryType();
551             displayNameTextView.setText(displayName);
552         }
553 
554         TextView countText = (TextView)view.findViewById(R.id.count);
555         if (directoryPartition.isLoading()) {
556             countText.setText(R.string.search_results_searching);
557         } else {
558             int count = cursor == null ? 0 : cursor.getCount();
559             if (directoryId != Directory.DEFAULT && directoryId != Directory.LOCAL_INVISIBLE
560                     && count >= getDirectoryResultLimit()) {
561                 countText.setText(mContext.getString(
562                         R.string.foundTooManyContacts, getDirectoryResultLimit()));
563             } else {
564                 countText.setText(getQuantityText(
565                         count, R.string.listFoundAllContactsZero, R.plurals.searchFoundContacts));
566             }
567         }
568     }
569 
570     /**
571      * Checks whether the contact entry at the given position represents the user's profile.
572      */
isUserProfile(int position)573     protected boolean isUserProfile(int position) {
574         // The profile only ever appears in the first position if it is present.  So if the position
575         // is anything beyond 0, it can't be the profile.
576         boolean isUserProfile = false;
577         if (position == 0) {
578             int partition = getPartitionForPosition(position);
579             if (partition >= 0) {
580                 // Save the old cursor position - the call to getItem() may modify the cursor
581                 // position.
582                 int offset = getCursor(partition).getPosition();
583                 Cursor cursor = (Cursor) getItem(position);
584                 if (cursor != null) {
585                     int profileColumnIndex = cursor.getColumnIndex(Contacts.IS_USER_PROFILE);
586                     if (profileColumnIndex != -1) {
587                         isUserProfile = cursor.getInt(profileColumnIndex) == 1;
588                     }
589                     // Restore the old cursor position.
590                     cursor.moveToPosition(offset);
591                 }
592             }
593         }
594         return isUserProfile;
595     }
596 
597     // TODO: fix PluralRules to handle zero correctly and use Resources.getQuantityText directly
getQuantityText(int count, int zeroResourceId, int pluralResourceId)598     public String getQuantityText(int count, int zeroResourceId, int pluralResourceId) {
599         if (count == 0) {
600             return getContext().getString(zeroResourceId);
601         } else {
602             String format = getContext().getResources()
603                     .getQuantityText(pluralResourceId, count).toString();
604             return String.format(format, count);
605         }
606     }
607 
isPhotoSupported(int partitionIndex)608     public boolean isPhotoSupported(int partitionIndex) {
609         Partition partition = getPartition(partitionIndex);
610         if (partition instanceof DirectoryPartition) {
611             return ((DirectoryPartition) partition).isPhotoSupported();
612         }
613         return true;
614     }
615 
616     /**
617      * Returns the currently selected filter.
618      */
getFilter()619     public ContactListFilter getFilter() {
620         return mFilter;
621     }
622 
setFilter(ContactListFilter filter)623     public void setFilter(ContactListFilter filter) {
624         mFilter = filter;
625     }
626 
627     // TODO: move sharable logic (bindXX() methods) to here with extra arguments
628 
629     /**
630      * Loads the photo for the quick contact view and assigns the contact uri.
631      * @param photoIdColumn Index of the photo id column
632      * @param photoUriColumn Index of the photo uri column. Optional: Can be -1
633      * @param contactIdColumn Index of the contact id column
634      * @param lookUpKeyColumn Index of the lookup key column
635      */
bindQuickContact(final ContactListItemView view, int partitionIndex, Cursor cursor, int photoIdColumn, int photoUriColumn, int contactIdColumn, int lookUpKeyColumn)636     protected void bindQuickContact(final ContactListItemView view, int partitionIndex,
637             Cursor cursor, int photoIdColumn, int photoUriColumn, int contactIdColumn,
638             int lookUpKeyColumn) {
639         long photoId = 0;
640         if (!cursor.isNull(photoIdColumn)) {
641             photoId = cursor.getLong(photoIdColumn);
642         }
643 
644         QuickContactBadge quickContact = view.getQuickContact();
645         quickContact.assignContactUri(
646                 getContactUri(partitionIndex, cursor, contactIdColumn, lookUpKeyColumn));
647 
648         if (photoId != 0 || photoUriColumn == -1) {
649             getPhotoLoader().loadThumbnail(quickContact, photoId, mDarkTheme);
650         } else {
651             final String photoUriString = cursor.getString(photoUriColumn);
652             final Uri photoUri = photoUriString == null ? null : Uri.parse(photoUriString);
653             getPhotoLoader().loadPhoto(quickContact, photoUri, -1, mDarkTheme);
654         }
655 
656     }
657 
getContactUri(int partitionIndex, Cursor cursor, int contactIdColumn, int lookUpKeyColumn)658     protected Uri getContactUri(int partitionIndex, Cursor cursor,
659             int contactIdColumn, int lookUpKeyColumn) {
660         long contactId = cursor.getLong(contactIdColumn);
661         String lookupKey = cursor.getString(lookUpKeyColumn);
662         Uri uri = Contacts.getLookupUri(contactId, lookupKey);
663         long directoryId = ((DirectoryPartition)getPartition(partitionIndex)).getDirectoryId();
664         if (directoryId != Directory.DEFAULT) {
665             uri = uri.buildUpon().appendQueryParameter(
666                     ContactsContract.DIRECTORY_PARAM_KEY, String.valueOf(directoryId)).build();
667         }
668         return uri;
669     }
670 
setContactsCount(String count)671     public void setContactsCount(String count) {
672         mContactsCount = count;
673     }
674 
getContactsCount()675     public String getContactsCount() {
676         return mContactsCount;
677     }
678 }
679