| /* |
| * Copyright (C) 2007 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.android.contacts; |
| |
| import static com.android.contacts.ContactEntryAdapter.AGGREGATE_PROJECTION; |
| import static com.android.contacts.ContactEntryAdapter.AGGREGATE_STARRED_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_1_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_2_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_3_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_4_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_5_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_9_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_CONTACT_ID_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_ID_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_IS_SUPER_PRIMARY_COLUMN; |
| import static com.android.contacts.ContactEntryAdapter.DATA_MIMETYPE_COLUMN; |
| |
| import com.android.contacts.SplitAggregateView.OnContactSelectedListener; |
| import com.android.internal.telephony.ITelephony; |
| |
| import android.app.AlertDialog; |
| import android.app.Dialog; |
| import android.app.ListActivity; |
| import android.content.ActivityNotFoundException; |
| import android.content.ContentResolver; |
| import android.content.ContentUris; |
| import android.content.ContentValues; |
| import android.content.Context; |
| import android.content.DialogInterface; |
| import android.content.Intent; |
| import android.content.DialogInterface.OnClickListener; |
| import android.content.pm.PackageManager; |
| import android.content.pm.ResolveInfo; |
| import android.content.res.Resources; |
| import android.database.ContentObserver; |
| import android.database.Cursor; |
| import android.graphics.Bitmap; |
| import android.graphics.drawable.Drawable; |
| import android.media.Ringtone; |
| import android.media.RingtoneManager; |
| import android.net.Uri; |
| import android.os.Bundle; |
| import android.os.Handler; |
| import android.os.RemoteException; |
| import android.os.ServiceManager; |
| import android.os.SystemClock; |
| import android.provider.ContactsContract.Aggregates; |
| import android.provider.ContactsContract.AggregationExceptions; |
| import android.provider.ContactsContract.CommonDataKinds; |
| import android.provider.ContactsContract.Data; |
| import android.text.TextUtils; |
| import android.util.Log; |
| import android.view.ContextMenu; |
| import android.view.ContextThemeWrapper; |
| import android.view.KeyEvent; |
| import android.view.Menu; |
| import android.view.MenuItem; |
| import android.view.View; |
| import android.view.ViewGroup; |
| import android.view.ContextMenu.ContextMenuInfo; |
| import android.widget.AdapterView; |
| import android.widget.CheckBox; |
| import android.widget.ImageView; |
| import android.widget.ListView; |
| import android.widget.TextView; |
| import android.widget.Toast; |
| |
| import java.util.ArrayList; |
| |
| /** |
| * Displays the details of a specific contact. |
| */ |
| public class ViewContactActivity extends ListActivity |
| implements View.OnCreateContextMenuListener, View.OnClickListener, |
| DialogInterface.OnClickListener { |
| private static final String TAG = "ViewContact"; |
| private static final String SHOW_BARCODE_INTENT = "com.google.zxing.client.android.ENCODE"; |
| |
| private static final boolean SHOW_SEPARATORS = false; |
| |
| private static final int DIALOG_CONFIRM_DELETE = 1; |
| |
| public static final int MENU_ITEM_DELETE = 1; |
| public static final int MENU_ITEM_MAKE_DEFAULT = 2; |
| public static final int MENU_ITEM_SHOW_BARCODE = 3; |
| public static final int MENU_ITEM_SPLIT_AGGREGATE = 4; |
| |
| private static final String[] AGGREGATION_EXCEPTIONS_PROJECTION = |
| new String[] { AggregationExceptions._ID}; |
| |
| private static final int AGGREGATION_EXCEPTIONS_COL_ID = 0; |
| |
| private Uri mUri; |
| private Uri mAggDataUri; |
| private ContentResolver mResolver; |
| private ViewAdapter mAdapter; |
| private int mNumPhoneNumbers = 0; |
| |
| /** |
| * A list of distinct contact IDs included in the current aggregate. |
| */ |
| private ArrayList<Long> mContactIds = new ArrayList<Long>(); |
| |
| /* package */ ArrayList<ViewEntry> mPhoneEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mSmsEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mEmailEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mPostalEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mImEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mOrganizationEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mGroupEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ViewEntry> mOtherEntries = new ArrayList<ViewEntry>(); |
| /* package */ ArrayList<ArrayList<ViewEntry>> mSections = new ArrayList<ArrayList<ViewEntry>>(); |
| |
| private Cursor mCursor; |
| private boolean mObserverRegistered; |
| |
| private ContentObserver mObserver = new ContentObserver(new Handler()) { |
| @Override |
| public boolean deliverSelfNotifications() { |
| return true; |
| } |
| |
| @Override |
| public void onChange(boolean selfChange) { |
| if (mCursor != null && !mCursor.isClosed()){ |
| dataChanged(); |
| } |
| } |
| }; |
| |
| public void onClick(DialogInterface dialog, int which) { |
| if (mCursor != null) { |
| if (mObserverRegistered) { |
| mCursor.unregisterContentObserver(mObserver); |
| mObserverRegistered = false; |
| } |
| mCursor.close(); |
| mCursor = null; |
| } |
| getContentResolver().delete(mUri, null, null); |
| finish(); |
| } |
| |
| public void onClick(View view) { |
| if (!mObserverRegistered) { |
| return; |
| } |
| switch (view.getId()) { |
| case R.id.star: { |
| mCursor.moveToFirst(); |
| int oldStarredState = mCursor.getInt(AGGREGATE_STARRED_COLUMN); |
| ContentValues values = new ContentValues(1); |
| values.put(Aggregates.STARRED, oldStarredState == 1 ? 0 : 1); |
| getContentResolver().update(mUri, values, null, null); |
| break; |
| } |
| } |
| } |
| |
| private TextView mNameView; |
| private TextView mPhoneticNameView; // may be null in some locales |
| private ImageView mPhotoView; |
| private int mNoPhotoResource; |
| private CheckBox mStarView; |
| private boolean mShowSmsLinksForAllPhones; |
| |
| @Override |
| protected void onCreate(Bundle icicle) { |
| super.onCreate(icicle); |
| |
| setContentView(R.layout.view_contact); |
| getListView().setOnCreateContextMenuListener(this); |
| |
| mNameView = (TextView) findViewById(R.id.name); |
| mPhoneticNameView = (TextView) findViewById(R.id.phonetic_name); |
| mPhotoView = (ImageView) findViewById(R.id.photo); |
| mStarView = (CheckBox) findViewById(R.id.star); |
| mStarView.setOnClickListener(this); |
| |
| // Set the photo with a random "no contact" image |
| long now = SystemClock.elapsedRealtime(); |
| int num = (int) now & 0xf; |
| if (num < 9) { |
| // Leaning in from right, common |
| mNoPhotoResource = R.drawable.ic_contact_picture; |
| } else if (num < 14) { |
| // Leaning in from left uncommon |
| mNoPhotoResource = R.drawable.ic_contact_picture_2; |
| } else { |
| // Coming in from the top, rare |
| mNoPhotoResource = R.drawable.ic_contact_picture_3; |
| } |
| |
| mUri = getIntent().getData(); |
| mAggDataUri = Uri.withAppendedPath(mUri, "data"); |
| mResolver = getContentResolver(); |
| |
| // Build the list of sections. The order they're added to mSections dictates the |
| // order they are displayed in the list. |
| mSections.add(mPhoneEntries); |
| mSections.add(mSmsEntries); |
| mSections.add(mEmailEntries); |
| mSections.add(mImEntries); |
| mSections.add(mPostalEntries); |
| mSections.add(mOrganizationEntries); |
| mSections.add(mGroupEntries); |
| mSections.add(mOtherEntries); |
| |
| //TODO Read this value from a preference |
| mShowSmsLinksForAllPhones = true; |
| |
| mCursor = mResolver.query(mAggDataUri, |
| AGGREGATE_PROJECTION, null, null, null); |
| } |
| |
| @Override |
| protected void onResume() { |
| super.onResume(); |
| mObserverRegistered = true; |
| mCursor.registerContentObserver(mObserver); |
| dataChanged(); |
| } |
| |
| @Override |
| protected void onPause() { |
| super.onPause(); |
| if (mCursor != null) { |
| if (mObserverRegistered) { |
| mObserverRegistered = false; |
| mCursor.unregisterContentObserver(mObserver); |
| } |
| mCursor.deactivate(); |
| } |
| } |
| |
| @Override |
| protected void onDestroy() { |
| super.onDestroy(); |
| |
| if (mCursor != null) { |
| if (mObserverRegistered) { |
| mCursor.unregisterContentObserver(mObserver); |
| mObserverRegistered = false; |
| } |
| mCursor.close(); |
| } |
| } |
| |
| @Override |
| protected Dialog onCreateDialog(int id) { |
| switch (id) { |
| case DIALOG_CONFIRM_DELETE: |
| return new AlertDialog.Builder(this) |
| .setTitle(R.string.deleteConfirmation_title) |
| .setIcon(android.R.drawable.ic_dialog_alert) |
| .setMessage(R.string.deleteConfirmation) |
| .setNegativeButton(android.R.string.cancel, null) |
| .setPositiveButton(android.R.string.ok, this) |
| .setCancelable(false) |
| .create(); |
| } |
| return null; |
| } |
| |
| private void dataChanged() { |
| mCursor.requery(); |
| if (mCursor.moveToFirst()) { |
| // Set the star |
| mStarView.setChecked(mCursor.getInt(AGGREGATE_STARRED_COLUMN) == 1 ? true : false); |
| |
| // Build up the contact entries |
| buildEntries(mCursor); |
| if (mAdapter == null) { |
| mAdapter = new ViewAdapter(this, mSections); |
| setListAdapter(mAdapter); |
| } else { |
| mAdapter.setSections(mSections, SHOW_SEPARATORS); |
| } |
| } else { |
| Toast.makeText(this, R.string.invalidContactMessage, Toast.LENGTH_SHORT).show(); |
| Log.e(TAG, "invalid contact uri: " + mUri); |
| finish(); |
| } |
| } |
| |
| @Override |
| public boolean onCreateOptionsMenu(Menu menu) { |
| menu.add(0, 0, 0, R.string.menu_editContact) |
| .setIcon(android.R.drawable.ic_menu_edit) |
| .setIntent(new Intent(Intent.ACTION_EDIT, mUri)) |
| .setAlphabeticShortcut('e'); |
| menu.add(0, MENU_ITEM_DELETE, 0, R.string.menu_deleteContact) |
| .setIcon(android.R.drawable.ic_menu_delete); |
| menu.add(0, MENU_ITEM_SPLIT_AGGREGATE, 0, R.string.menu_splitAggregate) |
| .setIcon(android.R.drawable.ic_menu_share); |
| return true; |
| } |
| |
| @Override |
| public boolean onPrepareOptionsMenu(Menu menu) { |
| super.onPrepareOptionsMenu(menu); |
| // Perform this check each time the menu is about to be shown, because the Barcode Scanner |
| // could be installed or uninstalled at any time. |
| if (isBarcodeScannerInstalled()) { |
| if (menu.findItem(MENU_ITEM_SHOW_BARCODE) == null) { |
| menu.add(0, MENU_ITEM_SHOW_BARCODE, 0, R.string.menu_showBarcode) |
| .setIcon(R.drawable.ic_menu_show_barcode); |
| } |
| } else { |
| menu.removeItem(MENU_ITEM_SHOW_BARCODE); |
| } |
| |
| boolean isAggregate = mContactIds.size() > 1; |
| menu.findItem(MENU_ITEM_SPLIT_AGGREGATE).setEnabled(isAggregate); |
| return true; |
| } |
| |
| private boolean isBarcodeScannerInstalled() { |
| final Intent intent = new Intent(SHOW_BARCODE_INTENT); |
| ResolveInfo ri = getPackageManager().resolveActivity(intent, |
| PackageManager.MATCH_DEFAULT_ONLY); |
| return ri != null; |
| } |
| |
| @Override |
| public void onCreateContextMenu(ContextMenu menu, View view, ContextMenuInfo menuInfo) { |
| AdapterView.AdapterContextMenuInfo info; |
| try { |
| info = (AdapterView.AdapterContextMenuInfo) menuInfo; |
| } catch (ClassCastException e) { |
| Log.e(TAG, "bad menuInfo", e); |
| return; |
| } |
| |
| // This can be null sometimes, don't crash... |
| if (info == null) { |
| Log.e(TAG, "bad menuInfo"); |
| return; |
| } |
| |
| ViewEntry entry = ContactEntryAdapter.getEntry(mSections, info.position, SHOW_SEPARATORS); |
| if (entry.mimetype.equals(CommonDataKinds.Phone.CONTENT_ITEM_TYPE)) { |
| menu.add(0, 0, 0, R.string.menu_call).setIntent(entry.intent); |
| menu.add(0, 0, 0, R.string.menu_sendSMS).setIntent(entry.auxIntent); |
| if (entry.primaryIcon == -1) { |
| menu.add(0, MENU_ITEM_MAKE_DEFAULT, 0, R.string.menu_makeDefaultNumber); |
| } |
| } else if (entry.mimetype.equals(CommonDataKinds.Email.CONTENT_ITEM_TYPE)) { |
| menu.add(0, 0, 0, R.string.menu_sendEmail).setIntent(entry.intent); |
| if (entry.primaryIcon == -1) { |
| menu.add(0, MENU_ITEM_MAKE_DEFAULT, 0, R.string.menu_makeDefaultEmail); |
| } |
| } else if (entry.mimetype.equals(CommonDataKinds.Postal.CONTENT_ITEM_TYPE)) { |
| menu.add(0, 0, 0, R.string.menu_viewAddress).setIntent(entry.intent); |
| } |
| // TODO(emillar): add back with group support. |
| /* else if (entry.mimetype.equals()) { |
| menu.add(0, 0, 0, R.string.menu_viewGroup).setIntent(entry.intent); |
| } */ |
| } |
| |
| @Override |
| public boolean onOptionsItemSelected(MenuItem item) { |
| switch (item.getItemId()) { |
| case MENU_ITEM_DELETE: { |
| // Get confirmation |
| showDialog(DIALOG_CONFIRM_DELETE); |
| return true; |
| } |
| |
| case MENU_ITEM_SPLIT_AGGREGATE: { |
| showSplitAggregateDialog(); |
| return true; |
| } |
| |
| // TODO(emillar) Bring this back. |
| /*case MENU_ITEM_SHOW_BARCODE: |
| if (mCursor.moveToFirst()) { |
| Intent intent = new Intent(SHOW_BARCODE_INTENT); |
| intent.putExtra("ENCODE_TYPE", "CONTACT_TYPE"); |
| Bundle bundle = new Bundle(); |
| String name = mCursor.getString(AGGREGATE_DISPLAY_NAME_COLUMN); |
| if (!TextUtils.isEmpty(name)) { |
| // Correctly handle when section headers are hidden |
| int sepAdjust = SHOW_SEPARATORS ? 1 : 0; |
| |
| bundle.putString(Contacts.Intents.Insert.NAME, name); |
| // The 0th ViewEntry in each ArrayList below is a separator item |
| int entriesToAdd = Math.min(mPhoneEntries.size() - sepAdjust, PHONE_KEYS.length); |
| for (int x = 0; x < entriesToAdd; x++) { |
| ViewEntry entry = mPhoneEntries.get(x + sepAdjust); |
| bundle.putString(PHONE_KEYS[x], entry.data); |
| } |
| entriesToAdd = Math.min(mEmailEntries.size() - sepAdjust, EMAIL_KEYS.length); |
| for (int x = 0; x < entriesToAdd; x++) { |
| ViewEntry entry = mEmailEntries.get(x + sepAdjust); |
| bundle.putString(EMAIL_KEYS[x], entry.data); |
| } |
| if (mPostalEntries.size() >= 1 + sepAdjust) { |
| ViewEntry entry = mPostalEntries.get(sepAdjust); |
| bundle.putString(Contacts.Intents.Insert.POSTAL, entry.data); |
| } |
| intent.putExtra("ENCODE_DATA", bundle); |
| try { |
| startActivity(intent); |
| } catch (ActivityNotFoundException e) { |
| // The check in onPrepareOptionsMenu() should make this impossible, but |
| // for safety I'm catching the exception rather than crashing. Ideally |
| // I'd call Menu.removeItem() here too, but I don't see a way to get |
| // the options menu. |
| Log.e(TAG, "Show barcode menu item was clicked but Barcode Scanner " + |
| "was not installed."); |
| } |
| return true; |
| } |
| } |
| break; */ |
| } |
| return super.onOptionsItemSelected(item); |
| } |
| |
| @Override |
| public boolean onContextItemSelected(MenuItem item) { |
| switch (item.getItemId()) { |
| case MENU_ITEM_MAKE_DEFAULT: { |
| if (makeItemDefault(item)) { |
| return true; |
| } |
| break; |
| } |
| } |
| |
| return super.onContextItemSelected(item); |
| } |
| |
| private boolean makeItemDefault(MenuItem item) { |
| ViewEntry entry = getViewEntryForMenuItem(item); |
| if (entry == null) { |
| return false; |
| } |
| |
| // Update the primary values in the data record. |
| ContentValues values = new ContentValues(2); |
| values.put(Data.IS_PRIMARY, 1); |
| values.put(Data.IS_SUPER_PRIMARY, 1); |
| |
| Log.i(TAG, mUri.toString()); |
| getContentResolver().update(ContentUris.withAppendedId(Data.CONTENT_URI, entry.id), |
| values, null, null); |
| dataChanged(); |
| return true; |
| } |
| |
| /** |
| * Shows a dialog that contains a list of all constituent contacts in this aggregate. |
| * The user picks a contact to be split into its own aggregate or clicks Cancel. |
| */ |
| private void showSplitAggregateDialog() { |
| |
| // Wrap this dialog in a specific theme so that list items have correct text color. |
| final ContextThemeWrapper dialogContext = |
| new ContextThemeWrapper(this, android.R.style.Theme_Light); |
| AlertDialog.Builder builder = |
| new AlertDialog.Builder(dialogContext); |
| builder.setTitle(getString(R.string.splitAggregate_title)); |
| |
| final SplitAggregateView view = new SplitAggregateView(dialogContext, mUri); |
| builder.setView(view); |
| |
| builder.setInverseBackgroundForced(true); |
| builder.setCancelable(true); |
| builder.setNegativeButton(android.R.string.cancel, |
| new OnClickListener() { |
| public void onClick(DialogInterface dialog, int which) { |
| dialog.dismiss(); |
| } |
| }); |
| final AlertDialog dialog = builder.create(); |
| |
| view.setOnContactSelectedListener(new OnContactSelectedListener() { |
| public void onContactSelected(long contactId) { |
| dialog.dismiss(); |
| splitContact(contactId); |
| } |
| }); |
| |
| dialog.show(); |
| } |
| |
| /** |
| * Given an ID of a constituent contact, splits it off into a separate aggregate. |
| */ |
| protected void splitContact(long contactToSplit) { |
| |
| ContentValues values = new ContentValues(3); |
| values.put(AggregationExceptions.AGGREGATE_ID, ContentUris.parseId(mUri)); |
| values.put(AggregationExceptions.CONTACT_ID, contactToSplit); |
| values.put(AggregationExceptions.TYPE, AggregationExceptions.TYPE_KEEP_OUT); |
| |
| mResolver.update(AggregationExceptions.CONTENT_URI, values, null, null); |
| |
| mAdapter.notifyDataSetChanged(); |
| } |
| |
| private ViewEntry getViewEntryForMenuItem(MenuItem item) { |
| AdapterView.AdapterContextMenuInfo info; |
| try { |
| info = (AdapterView.AdapterContextMenuInfo) item.getMenuInfo(); |
| } catch (ClassCastException e) { |
| Log.e(TAG, "bad menuInfo", e); |
| return null; |
| } |
| |
| return ContactEntryAdapter.getEntry(mSections, info.position, SHOW_SEPARATORS); |
| } |
| |
| @Override |
| public boolean onKeyDown(int keyCode, KeyEvent event) { |
| switch (keyCode) { |
| case KeyEvent.KEYCODE_CALL: { |
| try { |
| ITelephony phone = ITelephony.Stub.asInterface( |
| ServiceManager.checkService("phone")); |
| if (phone != null && !phone.isIdle()) { |
| // Skip out and let the key be handled at a higher level |
| break; |
| } |
| } catch (RemoteException re) { |
| // Fall through and try to call the contact |
| } |
| |
| int index = getListView().getSelectedItemPosition(); |
| if (index != -1) { |
| ViewEntry entry = ViewAdapter.getEntry(mSections, index, SHOW_SEPARATORS); |
| if (entry.intent.getAction() == Intent.ACTION_CALL_PRIVILEGED) { |
| startActivity(entry.intent); |
| } |
| } else if (mNumPhoneNumbers != 0) { |
| // There isn't anything selected, call the default number |
| Intent intent = new Intent(Intent.ACTION_CALL_PRIVILEGED, mUri); |
| startActivity(intent); |
| } |
| return true; |
| } |
| |
| case KeyEvent.KEYCODE_DEL: { |
| showDialog(DIALOG_CONFIRM_DELETE); |
| return true; |
| } |
| } |
| |
| return super.onKeyDown(keyCode, event); |
| } |
| |
| @Override |
| protected void onListItemClick(ListView l, View v, int position, long id) { |
| ViewEntry entry = ViewAdapter.getEntry(mSections, position, SHOW_SEPARATORS); |
| if (entry != null) { |
| Intent intent = entry.intent; |
| if (intent != null) { |
| try { |
| startActivity(intent); |
| } catch (ActivityNotFoundException e) { |
| Log.e(TAG, "No activity found for intent: " + intent); |
| signalError(); |
| } |
| } else { |
| signalError(); |
| } |
| } else { |
| signalError(); |
| } |
| } |
| |
| /** |
| * Signal an error to the user via a beep, or some other method. |
| */ |
| private void signalError() { |
| //TODO: implement this when we have the sonification APIs |
| } |
| |
| private Uri constructImToUrl(String host, String data) { |
| // don't encode the url, because the Activity Manager can't find using the encoded url |
| StringBuilder buf = new StringBuilder("imto://"); |
| buf.append(host); |
| buf.append('/'); |
| buf.append(data); |
| return Uri.parse(buf.toString()); |
| } |
| |
| /** |
| * Build up the entries to display on the screen. |
| * |
| * @param personCursor the URI for the contact being displayed |
| */ |
| private final void buildEntries(Cursor aggCursor) { |
| // Clear out the old entries |
| final int numSections = mSections.size(); |
| for (int i = 0; i < numSections; i++) { |
| mSections.get(i).clear(); |
| } |
| |
| mContactIds.clear(); |
| |
| // Build up method entries |
| if (mUri != null) { |
| Bitmap photoBitmap = null; |
| while (aggCursor.moveToNext()) { |
| final String mimetype = aggCursor.getString(DATA_MIMETYPE_COLUMN); |
| |
| ViewEntry entry = new ViewEntry(); |
| |
| final long id = aggCursor.getLong(DATA_ID_COLUMN); |
| final Uri uri = ContentUris.withAppendedId(Data.CONTENT_URI, id); |
| entry.id = id; |
| entry.uri = uri; |
| entry.mimetype = mimetype; |
| entry.contactId = aggCursor.getLong(DATA_CONTACT_ID_COLUMN); |
| if (!mContactIds.contains(entry.contactId)) { |
| mContactIds.add(entry.contactId); |
| } |
| |
| if (mimetype.equals(CommonDataKinds.Phone.CONTENT_ITEM_TYPE) |
| || mimetype.equals(CommonDataKinds.Email.CONTENT_ITEM_TYPE) |
| || mimetype.equals(CommonDataKinds.Postal.CONTENT_ITEM_TYPE) |
| || mimetype.equals(CommonDataKinds.Im.CONTENT_ITEM_TYPE)) { |
| final int type = aggCursor.getInt(DATA_1_COLUMN); |
| final String label = aggCursor.getString(DATA_3_COLUMN); |
| final String data = aggCursor.getString(DATA_2_COLUMN); |
| final boolean isSuperPrimary = "1".equals( |
| aggCursor.getString(DATA_IS_SUPER_PRIMARY_COLUMN)); |
| |
| // Don't crash if the data is bogus |
| if (TextUtils.isEmpty(data)) { |
| Log.w(TAG, "empty data for contact method " + id); |
| continue; |
| } |
| |
| // Build phone entries |
| if (mimetype.equals(CommonDataKinds.Phone.CONTENT_ITEM_TYPE)) { |
| mNumPhoneNumbers++; |
| |
| final CharSequence displayLabel = ContactsUtils.getDisplayLabel( |
| this, mimetype, type, label); |
| entry.label = buildActionString(R.string.actionCall, displayLabel, true); |
| entry.data = data; |
| entry.intent = new Intent(Intent.ACTION_CALL_PRIVILEGED, entry.uri); |
| entry.auxIntent = new Intent(Intent.ACTION_SENDTO, |
| Uri.fromParts("sms", data, null)); |
| if (isSuperPrimary) { |
| entry.primaryIcon = R.drawable.ic_default_number; |
| } |
| entry.actionIcon = android.R.drawable.sym_action_call; |
| mPhoneEntries.add(entry); |
| |
| if (type == CommonDataKinds.Phone.TYPE_MOBILE |
| || mShowSmsLinksForAllPhones) { |
| // Add an SMS entry |
| ViewEntry smsEntry = new ViewEntry(); |
| smsEntry.label = buildActionString( |
| R.string.actionText, displayLabel, true); |
| smsEntry.data = data; |
| smsEntry.id = id; |
| smsEntry.uri = uri; |
| smsEntry.intent = entry.auxIntent; |
| smsEntry.mimetype = FastTrackWindow.MIME_SMS_ADDRESS; |
| smsEntry.actionIcon = R.drawable.sym_action_sms; |
| mSmsEntries.add(smsEntry); |
| } |
| // Build email entries |
| } else if (mimetype.equals(CommonDataKinds.Email.CONTENT_ITEM_TYPE)) { |
| entry.label = buildActionString(R.string.actionEmail, |
| ContactsUtils.getDisplayLabel(this, mimetype, type, label), true); |
| entry.data = data; |
| entry.intent = new Intent(Intent.ACTION_SENDTO, |
| Uri.fromParts("mailto", data, null)); |
| entry.actionIcon = android.R.drawable.sym_action_email; |
| if (isSuperPrimary) { |
| entry.primaryIcon = R.drawable.ic_default_number; |
| } |
| mEmailEntries.add(entry); |
| // Build postal entries |
| } else if (mimetype.equals(CommonDataKinds.Postal.CONTENT_ITEM_TYPE)) { |
| entry.label = buildActionString(R.string.actionMap, |
| ContactsUtils.getDisplayLabel(this, mimetype, type, label), true); |
| entry.data = data; |
| entry.maxLines = 4; |
| entry.intent = new Intent(Intent.ACTION_VIEW, uri); |
| entry.actionIcon = R.drawable.sym_action_map; |
| mPostalEntries.add(entry); |
| // Build im entries |
| } else if (mimetype.equals(CommonDataKinds.Im.CONTENT_ITEM_TYPE)) { |
| String[] protocolStrings = getResources().getStringArray( |
| android.R.array.imProtocols); |
| Object protocolObj = ContactsUtils.decodeImProtocol( |
| aggCursor.getString(DATA_5_COLUMN)); |
| String host = null; |
| if (protocolObj instanceof Number) { |
| int protocol = ((Number) protocolObj).intValue(); |
| entry.label = buildActionString(R.string.actionChat, |
| protocolStrings[protocol], false); |
| host = ContactsUtils.lookupProviderNameFromId( |
| protocol).toLowerCase(); |
| if (protocol == CommonDataKinds.Im.PROTOCOL_GOOGLE_TALK |
| || protocol == CommonDataKinds.Im.PROTOCOL_MSN) { |
| entry.maxLabelLines = 2; |
| } |
| } else if (protocolObj != null) { |
| String providerName = (String) protocolObj; |
| entry.label = buildActionString(R.string.actionChat, |
| providerName, false); |
| host = providerName.toLowerCase(); |
| } |
| |
| // Only add the intent if there is a valid host |
| if (!TextUtils.isEmpty(host)) { |
| entry.intent = new Intent(Intent.ACTION_SENDTO, |
| constructImToUrl(host, data)); |
| } |
| entry.data = data; |
| //TODO(emillar) Do we want presence info? |
| /*if (!aggCursor.isNull(METHODS_STATUS_COLUMN)) { |
| entry.presenceIcon = Presence.getPresenceIconResourceId( |
| aggCursor.getInt(METHODS_STATUS_COLUMN)); |
| }*/ |
| entry.actionIcon = android.R.drawable.sym_action_chat; |
| mImEntries.add(entry); |
| } |
| // Set the name |
| } else if (mimetype.equals(CommonDataKinds.StructuredName.CONTENT_ITEM_TYPE)) { |
| String name = mCursor.getString(DATA_9_COLUMN); |
| if (TextUtils.isEmpty(name)) { |
| mNameView.setText(getText(android.R.string.unknownName)); |
| } else { |
| mNameView.setText(name); |
| } |
| /* |
| if (mPhoneticNameView != null) { |
| String phoneticName = mCursor.getString(CONTACT_PHONETIC_NAME_COLUMN); |
| mPhoneticNameView.setText(phoneticName); |
| } |
| */ |
| // Build organization entries |
| } else if (mimetype.equals(CommonDataKinds.Organization.CONTENT_ITEM_TYPE)) { |
| final String company = aggCursor.getString(DATA_3_COLUMN); |
| final String title = aggCursor.getString(DATA_4_COLUMN); |
| final boolean isPrimary = "1".equals(aggCursor.getString(DATA_5_COLUMN)); |
| |
| if (isPrimary) { |
| entry.primaryIcon = R.drawable.ic_default_number; |
| } |
| |
| // Don't crash if the data is bogus |
| if (TextUtils.isEmpty(company) && TextUtils.isEmpty(title)) { |
| Log.w(TAG, "empty data for contact method " + id); |
| continue; |
| } |
| |
| entry.data = title; |
| entry.actionIcon = R.drawable.sym_action_organization; |
| entry.label = company; |
| |
| mOrganizationEntries.add(entry); |
| // Build note entries |
| } else if (mimetype.equals(CommonDataKinds.Note.CONTENT_ITEM_TYPE)) { |
| entry.label = getString(R.string.label_notes); |
| entry.data = aggCursor.getString(DATA_1_COLUMN); |
| entry.id = 0; |
| entry.uri = null; |
| entry.intent = null; |
| entry.maxLines = 10; |
| entry.actionIcon = R.drawable.sym_note; |
| |
| if (TextUtils.isEmpty(entry.data)) { |
| Log.w(TAG, "empty data for contact method " + id); |
| continue; |
| } |
| |
| mOtherEntries.add(entry); |
| // Build the ringtone and send to voicemail entries |
| } else if (mimetype.equals(CommonDataKinds.CustomRingtone.CONTENT_ITEM_TYPE)) { |
| final Boolean sendToVoicemail = "1".equals(aggCursor.getString(DATA_1_COLUMN)); |
| final String ringtoneStr = aggCursor.getString(DATA_2_COLUMN); |
| |
| // TODO(emillar) we need to enforce uniqueness of custom ringtone entries on a |
| // single aggregate. This could be done by checking against the reference ID stored |
| // in the aggregate. |
| |
| // Build the send directly to voice mail entry |
| if (sendToVoicemail) { |
| entry.label = getString(R.string.actionIncomingCall); |
| entry.data = getString(R.string.detailIncomingCallsGoToVoicemail); |
| entry.actionIcon = R.drawable.sym_send_to_voicemail; |
| mOtherEntries.add(entry); |
| } else if (!TextUtils.isEmpty(ringtoneStr)) { |
| // Get the URI |
| Uri ringtoneUri = Uri.parse(ringtoneStr); |
| if (ringtoneUri != null) { |
| Ringtone ringtone = RingtoneManager.getRingtone(this, ringtoneUri); |
| if (ringtone != null) { |
| entry.label = getString(R.string.label_ringtone); |
| entry.data = ringtone.getTitle(this); |
| entry.uri = ringtoneUri; |
| entry.actionIcon = R.drawable.sym_ringtone; |
| mOtherEntries.add(entry); |
| } |
| } |
| } |
| // Load the photo |
| } else if (mimetype.equals(CommonDataKinds.Photo.CONTENT_ITEM_TYPE)) { |
| photoBitmap = ContactsUtils.loadContactPhoto(aggCursor, DATA_1_COLUMN, null); |
| } |
| |
| |
| // TODO(emillar) Add group entries |
| // // Build the group entries |
| // final Uri groupsUri = Uri.withAppendedPath(mUri, GroupMembership.CONTENT_DIRECTORY); |
| // Cursor groupCursor = mResolver.query(groupsUri, ContactsListActivity.GROUPS_PROJECTION, |
| // null, null, Groups.DEFAULT_SORT_ORDER); |
| // if (groupCursor != null) { |
| // try { |
| // StringBuilder sb = new StringBuilder(); |
| // |
| // while (groupCursor.moveToNext()) { |
| // String systemId = groupCursor.getString( |
| // ContactsListActivity.GROUPS_COLUMN_INDEX_SYSTEM_ID); |
| // |
| // if (systemId != null || Groups.GROUP_MY_CONTACTS.equals(systemId)) { |
| // continue; |
| // } |
| // |
| // String name = groupCursor.getString(ContactsListActivity.GROUPS_COLUMN_INDEX_NAME); |
| // if (!TextUtils.isEmpty(name)) { |
| // if (sb.length() == 0) { |
| // sb.append(name); |
| // } else { |
| // sb.append(getString(R.string.group_list, name)); |
| // } |
| // } |
| // } |
| // |
| // if (sb.length() > 0) { |
| // ViewEntry entry = new ViewEntry(); |
| // entry.kind = ContactEntryAdapter.Entry.KIND_GROUP; |
| // entry.label = getString(R.string.label_groups); |
| // entry.data = sb.toString(); |
| // entry.intent = new Intent(Intent.ACTION_EDIT, mUri); |
| // |
| // // TODO: Add an icon for the groups item. |
| // |
| // mGroupEntries.add(entry); |
| // } |
| // } finally { |
| // groupCursor.close(); |
| // } |
| // } |
| |
| } |
| |
| if (photoBitmap == null) { |
| photoBitmap = ContactsUtils.loadPlaceholderPhoto(mNoPhotoResource, this, null); |
| } |
| mPhotoView.setImageBitmap(photoBitmap); |
| } |
| } |
| |
| |
| String buildActionString(int actionResId, CharSequence type, boolean lowerCase) { |
| // If there is no type just display an empty string |
| if (type == null) { |
| type = ""; |
| } |
| |
| if (lowerCase) { |
| return getString(actionResId, type.toString().toLowerCase()); |
| } else { |
| return getString(actionResId, type.toString()); |
| } |
| } |
| |
| /** |
| * A basic structure with the data for a contact entry in the list. |
| */ |
| final static class ViewEntry extends ContactEntryAdapter.Entry { |
| public int primaryIcon = -1; |
| public Intent intent; |
| public Intent auxIntent = null; |
| public int presenceIcon = -1; |
| public int actionIcon = -1; |
| public int maxLabelLines = 1; |
| } |
| |
| private static final class ViewAdapter extends ContactEntryAdapter<ViewEntry> { |
| /** Cache of the children views of a row */ |
| static class ViewCache { |
| public TextView label; |
| public TextView data; |
| public ImageView actionIcon; |
| public ImageView presenceIcon; |
| |
| // Need to keep track of this too |
| ViewEntry entry; |
| } |
| |
| ViewAdapter(Context context, ArrayList<ArrayList<ViewEntry>> sections) { |
| super(context, sections, SHOW_SEPARATORS); |
| } |
| |
| @Override |
| public View getView(int position, View convertView, ViewGroup parent) { |
| ViewEntry entry = getEntry(mSections, position, false); |
| View v; |
| |
| ViewCache views; |
| |
| // Check to see if we can reuse convertView |
| if (convertView != null) { |
| v = convertView; |
| views = (ViewCache) v.getTag(); |
| } else { |
| // Create a new view if needed |
| v = mInflater.inflate(R.layout.list_item_text_icons, parent, false); |
| |
| // Cache the children |
| views = new ViewCache(); |
| views.label = (TextView) v.findViewById(android.R.id.text1); |
| views.data = (TextView) v.findViewById(android.R.id.text2); |
| views.actionIcon = (ImageView) v.findViewById(R.id.icon1); |
| views.presenceIcon = (ImageView) v.findViewById(R.id.icon2); |
| v.setTag(views); |
| } |
| |
| // Update the entry in the view cache |
| views.entry = entry; |
| |
| // Bind the data to the view |
| bindView(v, entry); |
| return v; |
| } |
| |
| @Override |
| protected View newView(int position, ViewGroup parent) { |
| // getView() handles this |
| throw new UnsupportedOperationException(); |
| } |
| |
| @Override |
| protected void bindView(View view, ViewEntry entry) { |
| final Resources resources = mContext.getResources(); |
| ViewCache views = (ViewCache) view.getTag(); |
| |
| // Set the label |
| TextView label = views.label; |
| setMaxLines(label, entry.maxLabelLines); |
| label.setText(entry.label); |
| |
| // Set the data |
| TextView data = views.data; |
| if (data != null) { |
| data.setText(entry.data); |
| setMaxLines(data, entry.maxLines); |
| } |
| |
| // Set the action icon |
| ImageView action = views.actionIcon; |
| if (entry.actionIcon != -1) { |
| action.setImageDrawable(resources.getDrawable(entry.actionIcon)); |
| action.setVisibility(View.VISIBLE); |
| } else { |
| // Things should still line up as if there was an icon, so make it invisible |
| action.setVisibility(View.INVISIBLE); |
| } |
| |
| // Set the presence icon |
| Drawable presenceIcon = null; |
| if (entry.primaryIcon != -1) { |
| presenceIcon = resources.getDrawable(entry.primaryIcon); |
| } else if (entry.presenceIcon != -1) { |
| presenceIcon = resources.getDrawable(entry.presenceIcon); |
| } |
| |
| ImageView presence = views.presenceIcon; |
| if (presenceIcon != null) { |
| presence.setImageDrawable(presenceIcon); |
| presence.setVisibility(View.VISIBLE); |
| } else { |
| presence.setVisibility(View.GONE); |
| } |
| } |
| |
| private void setMaxLines(TextView textView, int maxLines) { |
| if (maxLines == 1) { |
| textView.setSingleLine(true); |
| textView.setEllipsize(TextUtils.TruncateAt.END); |
| } else { |
| textView.setSingleLine(false); |
| textView.setMaxLines(maxLines); |
| textView.setEllipsize(null); |
| } |
| } |
| } |
| } |