@ -17,8 +17,12 @@ import org.thoughtcrime.securesms.contactshare.Contact.Name;
import org.thoughtcrime.securesms.contactshare.Contact.Phone ;
import org.thoughtcrime.securesms.contactshare.Contact.PostalAddress ;
import org.thoughtcrime.securesms.database.Address ;
import org.thoughtcrime.securesms.mms.PartAuthority ;
import org.thoughtcrime.securesms.providers.PersistentBlobProvider ;
import org.thoughtcrime.securesms.recipients.Recipient ;
import java.io.IOException ;
import java.io.InputStream ;
import java.util.ArrayList ;
import java.util.HashMap ;
import java.util.LinkedList ;
@ -26,6 +30,9 @@ import java.util.List;
import java.util.Map ;
import java.util.concurrent.Executor ;
import ezvcard.Ezvcard ;
import ezvcard.VCard ;
import static org.thoughtcrime.securesms.contactshare.Contact.* ;
public class ContactRepository {
@ -45,11 +52,18 @@ public class ContactRepository {
this . contactsDatabase = contactsDatabase ;
}
void getContacts ( @NonNull List < Long> contactId s, @NonNull ValueCallback < List < Contact > > callback ) {
void getContacts ( @NonNull List < Uri> contactUri s, @NonNull ValueCallback < List < Contact > > callback ) {
executor . execute ( ( ) - > {
List < Contact > contacts = new ArrayList < > ( contactIds . size ( ) ) ;
for ( long id : contactIds ) {
Contact contact = getContact ( id ) ;
List < Contact > contacts = new ArrayList < > ( contactUris . size ( ) ) ;
for ( Uri contactUri : contactUris ) {
Contact contact ;
if ( ContactsContract . AUTHORITY . equals ( contactUri . getAuthority ( ) ) ) {
contact = getContactFromSystemContacts ( ContactUtil . getContactIdFromUri ( contactUri ) ) ;
} else {
contact = getContactFromVcard ( contactUri ) ;
}
if ( contact ! = null ) {
contacts . add ( contact ) ;
}
@ -59,7 +73,7 @@ public class ContactRepository {
}
@WorkerThread
private @Nullable Contact getContact ( long contactId ) {
private @Nullable Contact getContact FromSystemContacts ( long contactId ) {
Name name = getName ( contactId ) ;
if ( name = = null ) {
Log . w ( TAG , "Couldn't find a name associated with the provided contact ID." ) ;
@ -73,6 +87,79 @@ public class ContactRepository {
return new Contact ( name , null , phoneNumbers , getEmails ( contactId ) , getPostalAddresses ( contactId ) , avatar ) ;
}
@WorkerThread
private @Nullable Contact getContactFromVcard ( @NonNull Uri uri ) {
Contact contact = null ;
try ( InputStream stream = PartAuthority . getAttachmentStream ( context , uri ) ) {
VCard vcard = Ezvcard . parse ( stream ) . first ( ) ;
ezvcard . property . StructuredName vName = vcard . getStructuredName ( ) ;
List < ezvcard . property . Telephone > vPhones = vcard . getTelephoneNumbers ( ) ;
List < ezvcard . property . Email > vEmails = vcard . getEmails ( ) ;
List < ezvcard . property . Address > vPostalAddresses = vcard . getAddresses ( ) ;
String organization = vcard . getOrganization ( ) ! = null & & ! vcard . getOrganization ( ) . getValues ( ) . isEmpty ( ) ? vcard . getOrganization ( ) . getValues ( ) . get ( 0 ) : null ;
String displayName = vcard . getFormattedName ( ) ! = null ? vcard . getFormattedName ( ) . getValue ( ) : null ;
if ( displayName = = null & & vName ! = null ) {
displayName = vName . getGiven ( ) ;
}
if ( displayName = = null & & vcard . getOrganization ( ) ! = null ) {
displayName = organization ;
}
if ( displayName = = null ) {
throw new IOException ( "No valid name." ) ;
}
Name name = new Name ( displayName ,
vName ! = null ? vName . getGiven ( ) : null ,
vName ! = null ? vName . getFamily ( ) : null ,
vName ! = null & & ! vName . getPrefixes ( ) . isEmpty ( ) ? vName . getPrefixes ( ) . get ( 0 ) : null ,
vName ! = null & & ! vName . getSuffixes ( ) . isEmpty ( ) ? vName . getSuffixes ( ) . get ( 0 ) : null ,
null ) ;
List < Phone > phoneNumbers = new ArrayList < > ( vPhones . size ( ) ) ;
for ( ezvcard . property . Telephone vEmail : vPhones ) {
String label = ! vEmail . getTypes ( ) . isEmpty ( ) ? getCleanedVcardType ( vEmail . getTypes ( ) . get ( 0 ) . getValue ( ) ) : null ;
phoneNumbers . add ( new Phone ( vEmail . getText ( ) , phoneTypeFromVcardType ( label ) , label ) ) ;
}
List < Email > emails = new ArrayList < > ( vEmails . size ( ) ) ;
for ( ezvcard . property . Email vEmail : vEmails ) {
String label = ! vEmail . getTypes ( ) . isEmpty ( ) ? getCleanedVcardType ( vEmail . getTypes ( ) . get ( 0 ) . getValue ( ) ) : null ;
emails . add ( new Email ( vEmail . getValue ( ) , emailTypeFromVcardType ( label ) , label ) ) ;
}
List < PostalAddress > postalAddresses = new ArrayList < > ( vPostalAddresses . size ( ) ) ;
for ( ezvcard . property . Address vPostalAddress : vPostalAddresses ) {
String label = ! vPostalAddress . getTypes ( ) . isEmpty ( ) ? getCleanedVcardType ( vPostalAddress . getTypes ( ) . get ( 0 ) . getValue ( ) ) : null ;
postalAddresses . add ( new PostalAddress ( postalAddressTypeFromVcardType ( label ) ,
label ,
vPostalAddress . getStreetAddress ( ) ,
vPostalAddress . getPoBox ( ) ,
null ,
vPostalAddress . getLocality ( ) ,
vPostalAddress . getRegion ( ) ,
vPostalAddress . getPostalCode ( ) ,
vPostalAddress . getCountry ( ) ) ) ;
}
contact = new Contact ( name , organization , phoneNumbers , emails , postalAddresses , null ) ;
} catch ( IOException e ) {
Log . w ( TAG , "Failed to parse the vcard." , e ) ;
}
if ( PersistentBlobProvider . AUTHORITY . equals ( uri . getAuthority ( ) ) ) {
PersistentBlobProvider . getInstance ( context ) . delete ( context , uri ) ;
}
return contact ;
}
@WorkerThread
private @Nullable Name getName ( long contactId ) {
try ( Cursor cursor = contactsDatabase . getNameDetails ( contactId ) ) {
@ -225,6 +312,13 @@ public class ContactRepository {
return Phone . Type . CUSTOM ;
}
private Phone . Type phoneTypeFromVcardType ( @Nullable String type ) {
if ( "home" . equalsIgnoreCase ( type ) ) return Phone . Type . HOME ;
else if ( "cell" . equalsIgnoreCase ( type ) ) return Phone . Type . MOBILE ;
else if ( "work" . equalsIgnoreCase ( type ) ) return Phone . Type . WORK ;
else return Phone . Type . CUSTOM ;
}
private Email . Type emailTypeFromContactType ( int type ) {
switch ( type ) {
case ContactsContract . CommonDataKinds . Email . TYPE_HOME :
@ -237,6 +331,13 @@ public class ContactRepository {
return Email . Type . CUSTOM ;
}
private Email . Type emailTypeFromVcardType ( @Nullable String type ) {
if ( "home" . equalsIgnoreCase ( type ) ) return Email . Type . HOME ;
else if ( "cell" . equalsIgnoreCase ( type ) ) return Email . Type . MOBILE ;
else if ( "work" . equalsIgnoreCase ( type ) ) return Email . Type . WORK ;
else return Email . Type . CUSTOM ;
}
private PostalAddress . Type postalAddressTypeFromContactType ( int type ) {
switch ( type ) {
case ContactsContract . CommonDataKinds . StructuredPostal . TYPE_HOME :
@ -247,6 +348,22 @@ public class ContactRepository {
return PostalAddress . Type . CUSTOM ;
}
private PostalAddress . Type postalAddressTypeFromVcardType ( @Nullable String type ) {
if ( "home" . equalsIgnoreCase ( type ) ) return PostalAddress . Type . HOME ;
else if ( "work" . equalsIgnoreCase ( type ) ) return PostalAddress . Type . WORK ;
else return PostalAddress . Type . CUSTOM ;
}
private String getCleanedVcardType ( @Nullable String type ) {
if ( TextUtils . isEmpty ( type ) ) return "" ;
if ( type . startsWith ( "x-" ) & & type . length ( ) > 2 ) {
return type . substring ( 2 ) ;
}
return type ;
}
interface ValueCallback < T > {
void onComplete ( @NonNull T value ) ;
}