BluetoothPbapVcardManager.java revision 51d7be27c1e6303ae9806842ff4f3766d3ce7623
1/* 2 * Copyright (c) 2008-2009, Motorola, Inc. 3 * 4 * All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions are met: 8 * 9 * - Redistributions of source code must retain the above copyright notice, 10 * this list of conditions and the following disclaimer. 11 * 12 * - Redistributions in binary form must reproduce the above copyright notice, 13 * this list of conditions and the following disclaimer in the documentation 14 * and/or other materials provided with the distribution. 15 * 16 * - Neither the name of the Motorola, Inc. nor the names of its contributors 17 * may be used to endorse or promote products derived from this software 18 * without specific prior written permission. 19 * 20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 21 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 23 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 24 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 25 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 26 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 27 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 30 * POSSIBILITY OF SUCH DAMAGE. 31 */ 32 33package com.android.bluetooth.pbap; 34 35import com.android.bluetooth.R; 36 37import android.net.Uri; 38import android.os.Handler; 39import android.text.TextUtils; 40import android.util.Log; 41import android.database.Cursor; 42import android.content.ContentResolver; 43import android.content.Context; 44import android.provider.CallLog; 45import android.provider.CallLog.Calls; 46import android.provider.ContactsContract.RawContacts; 47import android.provider.ContactsContract.PhoneLookup; 48import android.pim.vcard.VCardComposer; 49import android.pim.vcard.VCardConfig; 50import android.pim.vcard.VCardComposer.OneEntryHandler; 51import android.provider.ContactsContract; 52import android.provider.ContactsContract.CommonDataKinds; 53import android.provider.ContactsContract.Contacts; 54import android.provider.ContactsContract.Data; 55import android.provider.ContactsContract.CommonDataKinds.Phone; 56 57import javax.obex.ResponseCodes; 58import javax.obex.Operation; 59import java.io.IOException; 60import java.io.OutputStream; 61import java.io.Writer; 62import java.util.ArrayList; 63 64public class BluetoothPbapVcardManager { 65 private static final String TAG = "BluetoothPbapVcardManager"; 66 67 private static final boolean V = BluetoothPbapService.VERBOSE; 68 69 private ContentResolver mResolver; 70 71 private Context mContext; 72 73 private StringBuilder mVcardResults = null; 74 75 static final String[] PHONES_PROJECTION = new String[] { 76 Data._ID, // 0 77 CommonDataKinds.Phone.TYPE, // 1 78 CommonDataKinds.Phone.LABEL, // 2 79 CommonDataKinds.Phone.NUMBER, // 3 80 Contacts.DISPLAY_NAME, // 4 81 }; 82 83 private static final int ID_COLUMN_INDEX = 0; 84 85 private static final int PHONE_TYPE_COLUMN_INDEX = 1; 86 87 private static final int PHONE_LABEL_COLUMN_INDEX = 2; 88 89 private static final int PHONE_NUMBER_COLUMN_INDEX = 3; 90 91 private static final int CONTACTS_DISPLAY_NAME_COLUMN_INDEX = 4; 92 93 static final String SORT_ORDER_NAME = Contacts.DISPLAY_NAME + " ASC"; 94 95 static final String SORT_ORDER_PHONE_NUMBER = CommonDataKinds.Phone.NUMBER + " ASC"; 96 97 public BluetoothPbapVcardManager(final Context context) { 98 mContext = context; 99 mResolver = mContext.getContentResolver(); 100 } 101 102 public final String getOwnerPhoneNumberVcard(final boolean vcardType21) { 103 VCardComposer composer = new VCardComposer(mContext); 104 String name = BluetoothPbapService.getLocalPhoneName(); 105 String number = BluetoothPbapService.getLocalPhoneNum(); 106 String vcard = composer.composeVCardForPhoneOwnNumber(Phone.TYPE_MOBILE, name, number, 107 vcardType21); 108 return vcard; 109 } 110 111 public final int getPhonebookSize(final int type) { 112 int size; 113 switch (type) { 114 case BluetoothPbapObexServer.ContentType.PHONEBOOK: 115 size = getContactsSize(); 116 break; 117 default: 118 size = getCallHistorySize(type); 119 break; 120 } 121 if (V) Log.v(TAG, "getPhonebookSzie size = " + size + " type = " + type); 122 return size; 123 } 124 125 public final int getContactsSize() { 126 Uri myUri = RawContacts.CONTENT_URI; 127 int size = 0; 128 Cursor contactCursor = null; 129 try { 130 contactCursor = mResolver.query(myUri, null, null, null, null); 131 if (contactCursor != null) { 132 size = contactCursor.getCount() + 1; // always has the 0.vcf 133 } 134 } finally { 135 if (contactCursor != null) { 136 contactCursor.close(); 137 } 138 } 139 return size; 140 } 141 142 public final int getCallHistorySize(final int type) { 143 Uri myUri = CallLog.Calls.CONTENT_URI; 144 String selection = BluetoothPbapObexServer.createSelectionPara(type); 145 int size = 0; 146 Cursor callCursor = null; 147 try { 148 callCursor = mResolver.query(myUri, null, selection, null, 149 CallLog.Calls.DEFAULT_SORT_ORDER); 150 if (callCursor != null) { 151 size = callCursor.getCount(); 152 } 153 } finally { 154 if (callCursor != null) { 155 callCursor.close(); 156 } 157 } 158 return size; 159 } 160 161 public final ArrayList<String> loadCallHistoryList(final int type) { 162 Uri myUri = CallLog.Calls.CONTENT_URI; 163 String selection = BluetoothPbapObexServer.createSelectionPara(type); 164 String[] projection = new String[] { 165 Calls.NUMBER, Calls.CACHED_NAME 166 }; 167 final int CALLS_NUMBER_COLUMN_INDEX = 0; 168 final int CALLS_NAME_COLUMN_INDEX = 1; 169 170 Cursor callCursor = null; 171 ArrayList<String> list = new ArrayList<String>(); 172 try { 173 callCursor = mResolver.query(myUri, projection, selection, null, 174 CallLog.Calls.DEFAULT_SORT_ORDER); 175 if (callCursor != null) { 176 for (callCursor.moveToFirst(); !callCursor.isAfterLast(); 177 callCursor.moveToNext()) { 178 String name = callCursor.getString(CALLS_NAME_COLUMN_INDEX); 179 if (TextUtils.isEmpty(name)) { 180 // name not found,use number instead 181 name = callCursor.getString(CALLS_NUMBER_COLUMN_INDEX); 182 } 183 list.add(name); 184 } 185 } 186 } finally { 187 if (callCursor != null) { 188 callCursor.close(); 189 } 190 } 191 return list; 192 } 193 194 public final ArrayList<String> getPhonebookNameList() { 195 ArrayList<String> nameList = new ArrayList<String>(); 196 nameList.add(BluetoothPbapService.getLocalPhoneName()); 197 198 Uri myUri = Phone.CONTENT_URI; 199 Cursor phoneCursor = null; 200 try { 201 phoneCursor = mResolver.query(myUri, PHONES_PROJECTION, null, null, SORT_ORDER_NAME); 202 if (phoneCursor != null) { 203 for (phoneCursor.moveToFirst(); !phoneCursor.isAfterLast(); phoneCursor 204 .moveToNext()) { 205 String name = phoneCursor.getString(CONTACTS_DISPLAY_NAME_COLUMN_INDEX); 206 if (TextUtils.isEmpty(name)) { 207 name = mContext.getString(android.R.string.unknownName); 208 } 209 nameList.add(name); 210 } 211 } 212 } finally { 213 if (phoneCursor != null) { 214 phoneCursor.close(); 215 } 216 } 217 return nameList; 218 } 219 220 public final ArrayList<String> getPhonebookNumberList() { 221 ArrayList<String> numberList = new ArrayList<String>(); 222 numberList.add(BluetoothPbapService.getLocalPhoneNum()); 223 224 Uri myUri = Phone.CONTENT_URI; 225 Cursor phoneCursor = null; 226 try { 227 phoneCursor = mResolver.query(myUri, PHONES_PROJECTION, null, null, 228 SORT_ORDER_PHONE_NUMBER); 229 if (phoneCursor != null) { 230 for (phoneCursor.moveToFirst(); !phoneCursor.isAfterLast(); phoneCursor 231 .moveToNext()) { 232 String number = phoneCursor.getString(PHONE_NUMBER_COLUMN_INDEX); 233 if (TextUtils.isEmpty(number)) { 234 number = mContext.getString(R.string.defaultnumber); 235 } 236 numberList.add(number); 237 } 238 } 239 } finally { 240 if (phoneCursor != null) { 241 phoneCursor.close(); 242 } 243 } 244 return numberList; 245 } 246 247 public final int composeAndSendCallLogVcards(final int type, final Operation op, 248 final int startPoint, final int endPoint, final boolean vcardType21) { 249 if (startPoint < 1 || startPoint > endPoint) { 250 Log.e(TAG, "internal error: startPoint or endPoint is not correct."); 251 return ResponseCodes.OBEX_HTTP_INTERNAL_ERROR; 252 } 253 String typeSelection = BluetoothPbapObexServer.createSelectionPara(type); 254 255 Uri myUri = CallLog.Calls.CONTENT_URI; 256 final String[] CALLLOG_PROJECTION = new String[] { 257 CallLog.Calls._ID, // 0 258 }; 259 final int ID_COLUMN_INDEX = 0; 260 261 Cursor callsCursor = null; 262 long startPointId = 0; 263 long endPointId = 0; 264 try { 265 // Need test to see if order by _ID is ok here, or by date? 266 callsCursor = mResolver.query(myUri, CALLLOG_PROJECTION, typeSelection, null, 267 CallLog.Calls._ID); 268 if (callsCursor != null) { 269 callsCursor.moveToPosition(startPoint - 1); 270 startPointId = callsCursor.getLong(ID_COLUMN_INDEX); 271 if (V) Log.v(TAG, "Call Log query startPointId = " + startPointId); 272 if (startPoint == endPoint) { 273 endPointId = startPointId; 274 } else { 275 callsCursor.moveToPosition(endPoint - 1); 276 endPointId = callsCursor.getLong(ID_COLUMN_INDEX); 277 } 278 if (V) Log.v(TAG, "Call log query endPointId = " + endPointId); 279 } 280 } finally { 281 if (callsCursor != null) { 282 callsCursor.close(); 283 } 284 } 285 286 String recordSelection; 287 if (startPoint == endPoint) { 288 recordSelection = Calls._ID + "=" + startPointId; 289 } else { 290 recordSelection = Calls._ID + ">=" + startPointId + " AND " + Calls._ID + "<=" 291 + endPointId; 292 } 293 294 String selection; 295 if (typeSelection == null) { 296 selection = recordSelection; 297 } else { 298 selection = "(" + typeSelection + ") AND (" + recordSelection + ")"; 299 } 300 301 if (V) Log.v(TAG, "Call log query selection is: " + selection); 302 303 return composeAndSendVCards(op, selection, vcardType21, null, false); 304 } 305 306 public final int composeAndSendPhonebookVcards(final Operation op, final int startPoint, 307 final int endPoint, final boolean vcardType21, String ownerVCard) { 308 if (startPoint < 1 || startPoint > endPoint) { 309 Log.e(TAG, "internal error: startPoint or endPoint is not correct."); 310 return ResponseCodes.OBEX_HTTP_INTERNAL_ERROR; 311 } 312 Uri myUri = RawContacts.CONTENT_URI; 313 final String[] RAW_CONTACTS_PROJECTION = new String[] { 314 RawContacts._ID, // 0 315 }; 316 final int ID_COLUMN_INDEX = 0; 317 318 Cursor contactCursor = null; 319 long startPointId = 0; 320 long endPointId = 0; 321 try { 322 contactCursor = mResolver.query(myUri, RAW_CONTACTS_PROJECTION, null, null, 323 RawContacts._ID); 324 if (contactCursor != null) { 325 contactCursor.moveToPosition(startPoint - 1); 326 startPointId = contactCursor.getLong(ID_COLUMN_INDEX); 327 if (V) Log.v(TAG, "Query startPointId = " + startPointId); 328 if (startPoint == endPoint) { 329 endPointId = startPointId; 330 } else { 331 contactCursor.moveToPosition(endPoint - 1); 332 endPointId = contactCursor.getLong(ID_COLUMN_INDEX); 333 } 334 if (V) Log.v(TAG, "Query endPointId = " + endPointId); 335 } 336 } finally { 337 if (contactCursor != null) { 338 contactCursor.close(); 339 } 340 } 341 342 String selection; 343 if (startPoint == endPoint) { 344 selection = RawContacts._ID + "=" + startPointId; 345 } else { 346 selection = RawContacts._ID + ">=" + startPointId + " AND " + RawContacts._ID + "<=" 347 + endPointId; 348 } 349 350 if (V) Log.v(TAG, "Query selection is: " + selection); 351 352 return composeAndSendVCards(op, selection, vcardType21, ownerVCard, true); 353 } 354 355 public final int composeAndSendVCards(final Operation op, final String selection, 356 final boolean vcardType21, String ownerVCard, boolean isContacts) { 357 long timestamp = 0; 358 if (V) timestamp = System.currentTimeMillis(); 359 360 VCardComposer composer = null; 361 try { 362 // Currently only support Generic Vcard 2.1 and 3.0 363 int vcardType; 364 if (vcardType21) { 365 vcardType = VCardConfig.VCARD_TYPE_V21_GENERIC_UTF8; 366 } else { 367 vcardType = VCardConfig.VCARD_TYPE_V30_GENERIC_UTF8; 368 } 369 370 composer = new VCardComposer(mContext, vcardType, true); 371 composer.addHandler(new HandlerForStringBuffer(op, ownerVCard)); 372 final Uri contentUri = (isContacts ? Contacts.CONTENT_URI : CallLog.Calls.CONTENT_URI); 373 if (!composer.init(contentUri, selection, null, null)) { 374 return ResponseCodes.OBEX_HTTP_INTERNAL_ERROR; 375 } 376 377 while (!composer.isAfterLast()) { 378 if (!composer.createOneEntry()) { 379 Log.e(TAG, "Failed to read a contact. Error reason: " 380 + composer.getErrorReason()); 381 return ResponseCodes.OBEX_HTTP_INTERNAL_ERROR; 382 } 383 } 384 } finally { 385 if (composer != null) { 386 composer.terminate(); 387 } 388 } 389 390 if (V) Log.v(TAG, "Total vcard composing and sending out takes " 391 + (System.currentTimeMillis() - timestamp) + " ms"); 392 393 return ResponseCodes.OBEX_HTTP_OK; 394 } 395 396 /** 397 * Handler to emit VCard String to PCE once size grow to maxPacketSize. 398 */ 399 public class HandlerForStringBuffer implements OneEntryHandler { 400 @SuppressWarnings("hiding") 401 private Operation operation; 402 403 private OutputStream outputStream; 404 405 private int maxPacketSize; 406 407 private String phoneOwnVCard = null; 408 409 public HandlerForStringBuffer(Operation op, String ownerVCard) { 410 operation = op; 411 maxPacketSize = operation.getMaxPacketSize(); 412 if (V) Log.v(TAG, "getMaxPacketSize() = " + maxPacketSize); 413 if (ownerVCard != null) { 414 phoneOwnVCard = ownerVCard; 415 if (V) Log.v(TAG, "phone own number vcard:"); 416 if (V) Log.v(TAG, phoneOwnVCard); 417 } 418 } 419 420 public boolean onInit(Context context) { 421 try { 422 outputStream = operation.openOutputStream(); 423 mVcardResults = new StringBuilder(); 424 if (phoneOwnVCard != null) { 425 mVcardResults.append(phoneOwnVCard); 426 } 427 } catch (IOException e) { 428 Log.e(TAG, "open outputstrem failed" + e.toString()); 429 return false; 430 } 431 if (V) Log.v(TAG, "openOutputStream() ok."); 432 return true; 433 } 434 435 public boolean onEntryCreated(String vcard) { 436 int vcardLen = vcard.length(); 437 if (V) Log.v(TAG, "The length of this vcard is: " + vcardLen); 438 439 mVcardResults.append(vcard); 440 int vcardStringLen = mVcardResults.toString().length(); 441 if (V) Log.v(TAG, "The length of this vcardResults is: " + vcardStringLen); 442 443 if (vcardStringLen >= maxPacketSize) { 444 long timestamp = 0; 445 int position = 0; 446 447 // Need while loop to handle the big vcard case 448 while (position < (vcardStringLen - maxPacketSize)) { 449 if (V) timestamp = System.currentTimeMillis(); 450 451 String subStr = mVcardResults.toString().substring(position, 452 position + maxPacketSize); 453 try { 454 outputStream.write(subStr.getBytes(), 0, maxPacketSize); 455 } catch (IOException e) { 456 Log.e(TAG, "write outputstrem failed" + e.toString()); 457 return false; 458 } 459 if (V) Log.v(TAG, "Sending vcard String " + maxPacketSize + " bytes took " 460 + (System.currentTimeMillis() - timestamp) + " ms"); 461 462 position += maxPacketSize; 463 } 464 mVcardResults.delete(0, position); 465 } 466 return true; 467 } 468 469 public void onTerminate() { 470 // Send out last packet 471 String lastStr = mVcardResults.toString(); 472 try { 473 outputStream.write(lastStr.getBytes(), 0, lastStr.length()); 474 } catch (IOException e) { 475 Log.e(TAG, "write outputstrem failed" + e.toString()); 476 } 477 if (V) Log.v(TAG, "Last packet sent out, sending process complete!"); 478 479 if (!BluetoothPbapObexServer.closeStream(outputStream, operation)) { 480 if (V) Log.v(TAG, "CloseStream failed!"); 481 } else { 482 if (V) Log.v(TAG, "CloseStream ok!"); 483 } 484 } 485 } 486} 487