Advertisement
minafaw3

Contacts

Apr 1st, 2016
306
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 14.16 KB | None | 0 0
  1. package com.fsck.k9.helper;
  2.  
  3. import android.content.ContentResolver;
  4. import android.content.ContentUris;
  5. import android.content.Context;
  6. import android.database.Cursor;
  7. import android.net.Uri;
  8. import android.content.Intent;
  9. import android.provider.ContactsContract;
  10. import android.util.Log;
  11.  
  12. import com.fsck.k9.K9;
  13. import com.fsck.k9.mail.Address;
  14.  
  15. import java.util.ArrayList;
  16. import java.util.List;
  17.  
  18. /**
  19. * Helper class to access the contacts stored on the device.
  20. */
  21. public class Contacts {
  22. /**
  23. * The order in which the search results are returned by
  24. * {@link #searchContacts(CharSequence)}.
  25. */
  26. protected static final String SORT_ORDER =
  27. ContactsContract.CommonDataKinds.Email.TIMES_CONTACTED + " DESC, " +
  28. ContactsContract.Contacts.DISPLAY_NAME + ", " +
  29. ContactsContract.CommonDataKinds.Email._ID;
  30.  
  31. /**
  32. * Array of columns to load from the database.
  33. *
  34. * Important: The _ID field is needed by
  35. * {@link com.fsck.k9.EmailAddressAdapter} or more specificly by
  36. * {@link android.widget.ResourceCursorAdapter}.
  37. */
  38. protected static final String PROJECTION[] = {
  39. ContactsContract.CommonDataKinds.Email._ID,
  40. ContactsContract.Contacts.DISPLAY_NAME,
  41. ContactsContract.CommonDataKinds.Email.DATA,
  42. ContactsContract.CommonDataKinds.Email.CONTACT_ID
  43. };
  44.  
  45. /**
  46. * Index of the name field in the projection. This must match the order in
  47. * {@link #PROJECTION}.
  48. */
  49. protected static final int NAME_INDEX = 1;
  50.  
  51. /**
  52. * Index of the email address field in the projection. This must match the
  53. * order in {@link #PROJECTION}.
  54. */
  55. protected static final int EMAIL_INDEX = 2;
  56.  
  57. /**
  58. * Index of the contact id field in the projection. This must match the order in
  59. * {@link #PROJECTION}.
  60. */
  61. protected static final int CONTACT_ID_INDEX = 3;
  62.  
  63.  
  64. /**
  65. * Get instance of the Contacts class.
  66. *
  67. * <p>Note: This is left over from the days when we needed to have SDK-specific code to access
  68. * the contacts.</p>
  69. *
  70. * @param context A {@link Context} instance.
  71. * @return Appropriate {@link Contacts} instance for this device.
  72. */
  73. public static Contacts getInstance(Context context) {
  74. return new Contacts(context);
  75. }
  76.  
  77.  
  78. protected Context mContext;
  79. protected ContentResolver mContentResolver;
  80. protected Boolean mHasContactPicker;
  81.  
  82. /**
  83. * Constructor
  84. *
  85. * @param context A {@link Context} instance.
  86. */
  87. protected Contacts(Context context) {
  88. mContext = context;
  89. mContentResolver = context.getContentResolver();
  90. }
  91.  
  92. /**
  93. * Start the activity to add information to an existing contact or add a
  94. * new one.
  95. *
  96. * @param email An {@link Address} instance containing the email address
  97. * of the entity you want to add to the contacts. Optionally
  98. * the instance also contains the (display) name of that
  99. * entity.
  100. */
  101. public void createContact(final Address email) {
  102. final Uri contactUri = Uri.fromParts("mailto", email.getAddress(), null);
  103.  
  104. final Intent contactIntent = new Intent(ContactsContract.Intents.SHOW_OR_CREATE_CONTACT);
  105. contactIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  106. contactIntent.setData(contactUri);
  107.  
  108. // Pass along full E-mail string for possible create dialog
  109. contactIntent.putExtra(ContactsContract.Intents.EXTRA_CREATE_DESCRIPTION,
  110. email.toString());
  111.  
  112. // Only provide personal name hint if we have one
  113. final String senderPersonal = email.getPersonal();
  114. if (senderPersonal != null) {
  115. contactIntent.putExtra(ContactsContract.Intents.Insert.NAME, senderPersonal);
  116. }
  117.  
  118. mContext.startActivity(contactIntent);
  119. }
  120.  
  121. /**
  122. * Start the activity to add a phone number to an existing contact or add a new one.
  123. *
  124. * @param phoneNumber
  125. * The phone number to add to a contact, or to use when creating a new contact.
  126. */
  127. public void addPhoneContact(final String phoneNumber) {
  128. Intent addIntent = new Intent(Intent.ACTION_INSERT_OR_EDIT);
  129. addIntent.setType(ContactsContract.Contacts.CONTENT_ITEM_TYPE);
  130. addIntent.putExtra(ContactsContract.Intents.Insert.PHONE, Uri.decode(phoneNumber));
  131. addIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  132. mContext.startActivity(addIntent);
  133. }
  134.  
  135. /**
  136. * Check whether the provided email address belongs to one of the contacts.
  137. *
  138. * @param emailAddress The email address to look for.
  139. * @return <tt>true</tt>, if the email address belongs to a contact.
  140. * <tt>false</tt>, otherwise.
  141. */
  142. public boolean isInContacts(final String emailAddress) {
  143. boolean result = false;
  144.  
  145. final Cursor c = getContactByAddress(emailAddress);
  146.  
  147. if (c != null) {
  148. if (c.getCount() > 0) {
  149. result = true;
  150. }
  151. c.close();
  152. }
  153.  
  154. return result;
  155. }
  156.  
  157. /**
  158. * Filter the contacts matching the given search term.
  159. *
  160. * @param constraint The search term to filter the contacts.
  161. * @return A {@link Cursor} instance that can be used to get the
  162. * matching contacts.
  163. */
  164. public Cursor searchContacts(final CharSequence constraint) {
  165. final String filter = (constraint == null) ? "" : constraint.toString();
  166. final Uri uri = Uri.withAppendedPath(ContactsContract.CommonDataKinds.Email.CONTENT_FILTER_URI, Uri.encode(filter));
  167. final Cursor c = mContentResolver.query(
  168. uri,
  169. PROJECTION,
  170. null,
  171. null,
  172. SORT_ORDER);
  173.  
  174. if (c != null) {
  175. /*
  176. * To prevent expensive execution in the UI thread:
  177. * Cursors get lazily executed, so if you don't call anything on
  178. * the cursor before returning it from the background thread you'll
  179. * have a complied program for the cursor, but it won't have been
  180. * executed to generate the data yet. Often the execution is more
  181. * expensive than the compilation...
  182. */
  183. c.getCount();
  184. }
  185.  
  186. return c;
  187. }
  188.  
  189. /**
  190. * Get the name of the contact an email address belongs to.
  191. *
  192. * @param address The email address to search for.
  193. * @return The name of the contact the email address belongs to. Or
  194. * <tt>null</tt> if there's no matching contact.
  195. */
  196. public String getNameForAddress(String address) {
  197. if (address == null) {
  198. return null;
  199. }
  200.  
  201. final Cursor c = getContactByAddress(address);
  202.  
  203. String name = null;
  204. if (c != null) {
  205. if (c.getCount() > 0) {
  206. c.moveToFirst();
  207. name = getName(c);
  208. }
  209. c.close();
  210. }
  211.  
  212. return name;
  213. }
  214.  
  215. /**
  216. * Extract the name from a {@link Cursor} instance returned by
  217. * {@link #searchContacts(CharSequence)}.
  218. *
  219. * @param cursor The {@link Cursor} instance.
  220. * @return The name of the contact in the {@link Cursor}'s current row.
  221. */
  222. public String getName(Cursor cursor) {
  223. return cursor.getString(NAME_INDEX);
  224. }
  225.  
  226. /**
  227. * Extract the email address from a {@link Cursor} instance returned by
  228. * {@link #searchContacts(CharSequence)}.
  229. *
  230. * @param cursor The {@link Cursor} instance.
  231. * @return The email address of the contact in the {@link Cursor}'s current
  232. * row.
  233. */
  234. public String getEmail(Cursor cursor) {
  235. return cursor.getString(EMAIL_INDEX);
  236. }
  237.  
  238. /**
  239. * Mark contacts with the provided email addresses as contacted.
  240. *
  241. * @param addresses Array of {@link Address} objects describing the
  242. * contacts to be marked as contacted.
  243. */
  244. public void markAsContacted(final Address[] addresses) {
  245. //TODO: Optimize! Potentially a lot of database queries
  246. for (final Address address : addresses) {
  247. final Cursor c = getContactByAddress(address.getAddress());
  248.  
  249. if (c != null) {
  250. if (c.getCount() > 0) {
  251. c.moveToFirst();
  252. final long personId = c.getLong(CONTACT_ID_INDEX);
  253. ContactsContract.Contacts.markAsContacted(mContentResolver, personId);
  254. }
  255. c.close();
  256. }
  257. }
  258. }
  259.  
  260. /**
  261. * Creates the intent necessary to open a contact picker.
  262. *
  263. * @return The intent necessary to open a contact picker.
  264. */
  265. public Intent contactPickerIntent() {
  266. return new Intent(Intent.ACTION_PICK, ContactsContract.Contacts.CONTENT_URI);
  267. }
  268.  
  269. /**
  270. * Given a contact picker intent, returns a {@code ContactItem} instance for that contact.
  271. *
  272. * @param intent
  273. * The {@link Intent} returned by the contact picker.
  274. *
  275. * @return A {@link ContactItem} instance describing the picked contact. Or {@code null} if the
  276. * contact doesn't have any email addresses.
  277. */
  278. public ContactItem extractInfoFromContactPickerIntent(final Intent intent) {
  279. Cursor cursor = null;
  280. List<String> email = new ArrayList<String>();
  281.  
  282. try {
  283. Uri result = intent.getData();
  284. String displayName = null;
  285.  
  286. // Get the contact id from the Uri
  287. String id = result.getLastPathSegment();
  288.  
  289. cursor = mContentResolver.query(ContactsContract.CommonDataKinds.Email.CONTENT_URI, PROJECTION,
  290. ContactsContract.CommonDataKinds.Email.CONTACT_ID + "=?", new String[] { id }, null);
  291.  
  292. if (cursor != null) {
  293. while (cursor.moveToNext()) {
  294. String address = cursor.getString(EMAIL_INDEX);
  295. if (address != null) {
  296. email.add(address);
  297. }
  298.  
  299. if (displayName == null) {
  300. displayName = cursor.getString(NAME_INDEX);
  301. }
  302. }
  303.  
  304. // Return 'null' if no email addresses have been found
  305. if (email.isEmpty()) {
  306. return null;
  307. }
  308.  
  309. // Use the first email address found as display name
  310. if (displayName == null) {
  311. displayName = email.get(0);
  312. }
  313.  
  314. return new ContactItem(displayName, email);
  315. }
  316. } catch (Exception e) {
  317. Log.e(K9.LOG_TAG, "Failed to get email data", e);
  318. } finally {
  319. Utility.closeQuietly(cursor);
  320. }
  321.  
  322. return null;
  323. }
  324.  
  325. /**
  326. * Get URI to the picture of the contact with the supplied email address.
  327. *
  328. * @param address
  329. * An email address. The contact database is searched for a contact with this email
  330. * address.
  331. *
  332. * @return URI to the picture of the contact with the supplied email address. {@code null} if
  333. * no such contact could be found or the contact doesn't have a picture.
  334. */
  335. public Uri getPhotoUri(String address) {
  336. Long contactId;
  337. try {
  338. final Cursor c = getContactByAddress(address);
  339. if (c == null) {
  340. return null;
  341. }
  342.  
  343. try {
  344. if (!c.moveToFirst()) {
  345. return null;
  346. }
  347.  
  348. contactId = c.getLong(CONTACT_ID_INDEX);
  349. } finally {
  350. c.close();
  351. }
  352.  
  353. Cursor cur = mContentResolver.query(
  354. ContactsContract.Data.CONTENT_URI,
  355. null,
  356. ContactsContract.Data.CONTACT_ID + "=" + contactId + " AND "
  357. + ContactsContract.Data.MIMETYPE + "='"
  358. + ContactsContract.CommonDataKinds.Photo.CONTENT_ITEM_TYPE + "'", null,
  359. null);
  360. if (cur == null) {
  361. return null;
  362. }
  363. if (!cur.moveToFirst()) {
  364. cur.close();
  365. return null; // no photo
  366. }
  367. // Ok, they have a photo
  368. cur.close();
  369. Uri person = ContentUris.withAppendedId(ContactsContract.Contacts.CONTENT_URI, contactId);
  370. return Uri.withAppendedPath(person, ContactsContract.Contacts.Photo.CONTENT_DIRECTORY);
  371. } catch (Exception e) {
  372. Log.e(K9.LOG_TAG, "Couldn't fetch photo for contact with email " + address, e);
  373. return null;
  374. }
  375. }
  376.  
  377. /**
  378. * Does the device actually have a Contacts application suitable for
  379. * picking a contact. As hard as it is to believe, some vendors ship
  380. * without it.
  381. *
  382. * @return True, if the device supports picking contacts. False, otherwise.
  383. */
  384. public boolean hasContactPicker() {
  385. if (mHasContactPicker == null) {
  386. mHasContactPicker = !(mContext.getPackageManager().
  387. queryIntentActivities(contactPickerIntent(), 0).isEmpty());
  388. }
  389. return mHasContactPicker;
  390. }
  391.  
  392. /**
  393. * Return a {@link Cursor} instance that can be used to fetch information
  394. * about the contact with the given email address.
  395. *
  396. * @param address The email address to search for.
  397. * @return A {@link Cursor} instance that can be used to fetch information
  398. * about the contact with the given email address
  399. */
  400. private Cursor getContactByAddress(final String address) {
  401. final Uri uri = Uri.withAppendedPath(ContactsContract.CommonDataKinds.Email.CONTENT_LOOKUP_URI, Uri.encode(address));
  402. final Cursor c = mContentResolver.query(
  403. uri,
  404. PROJECTION,
  405. null,
  406. null,
  407. SORT_ORDER);
  408. return c;
  409. }
  410.  
  411. }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement