21
21
* along with this program. If not, see <http://www.gnu.org/licenses/>.
24
package am.ed.importcontacts;
24
package org.waxworlds.edam.importcontacts;
26
26
import java.util.HashMap;
27
27
import java.util.HashSet;
28
import java.util.Locale;
29
import org.waxworlds.edam.importcontacts.Importer.AbortImportException;
31
import android.app.Activity;
32
import android.database.Cursor;
33
import android.provider.Contacts;
30
36
public class ContactsCache
33
* A thing that can be used to identify (or lookup) a contact within the
34
* contacts cache. It is not a reference to a cache entry and may not
35
* identify an existing contact in the cache.
37
public static class CacheIdentifier
40
NONE, NAME, ORGANISATION, PRIMARY_NUMBER, PRIMARY_EMAIL }
43
private String _detail;
45
protected CacheIdentifier()
51
* Obtain a cache identifier. This routine is designed to be as robust
52
* as possible (in terms of bad or null detail values), and to return
53
* null when a cache identifier can not be created.
54
* @param type the detail type
55
* @param detail the detail
56
* @return the cache identifier, or null
58
public static CacheIdentifier factory( Type type, String detail )
62
case NAME: detail = normaliseName( detail ); break;
63
case ORGANISATION: detail = normaliseOrganisation( detail ); break;
64
case PRIMARY_NUMBER: detail = normalisePhoneNumber( detail ); break;
65
case PRIMARY_EMAIL: detail = normaliseEmailAddress( detail ); break;
68
if( detail == null ) return null;
69
return new CacheIdentifier( type, detail );
73
* Obtain a cache identifier from contact data. This routine is
74
* designed to be as robust as possible and may return null when a cache
75
* identifier can not be created.
76
* @param contact the contact data
77
* @return the cache identifier, or null
79
public static CacheIdentifier factory( Importer.ContactData contact )
81
CacheIdentifier ret = null;
83
if( contact.hasName() )
84
ret = factory( CacheIdentifier.Type.NAME,
86
if( ret == null && contact.hasPrimaryOrganisation() )
87
ret = factory( CacheIdentifier.Type.ORGANISATION,
88
contact.getPrimaryOrganisation() );
89
if( ret == null && contact.hasPrimaryNumber() )
90
ret = factory( CacheIdentifier.Type.PRIMARY_NUMBER,
91
contact.getPrimaryNumber() );
92
if( ret == null && contact.hasPrimaryEmail() )
93
ret = factory( CacheIdentifier.Type.PRIMARY_EMAIL,
94
contact.getPrimaryEmail() );
99
protected CacheIdentifier( Type type, String detail )
105
public Type getType()
110
public String getDetail()
116
// mappings of contact names, organisations and primary numbers to ids
117
private HashMap< String, Long > _contactsByName
118
= new HashMap< String, Long >();
119
private HashMap< String, Long > _contactsByOrg
120
= new HashMap< String, Long >();
121
private HashMap< String, Long > _contactsByNumber
122
= new HashMap< String, Long >();
123
private HashMap< String, Long > _contactsByEmail
124
= new HashMap< String, Long >();
126
// mapping of contact ids to sets of associated data
127
private HashMap< Long, HashSet< String > > _contactNumbers
128
= new HashMap< Long, HashSet< String > >();
129
private HashMap< Long, HashSet< String > > _contactEmails
130
= new HashMap< Long, HashSet< String > >();
131
private HashMap< Long, HashSet< String > > _contactAddresses
132
= new HashMap< Long, HashSet< String > >();
133
private HashMap< Long, HashSet< String > > _contactOrganisations
134
= new HashMap< Long, HashSet< String > >();
135
private HashMap< Long, HashSet< String > > _contactNotes
136
= new HashMap< Long, HashSet< String > >();
138
public boolean canLookup( CacheIdentifier identifier )
140
return lookup( identifier ) != null;
144
* Retrieve the contact id of a contact identified by the specified cache
145
* identifier, if it exists.
146
* @param identifier the cache identifier
147
* @return a contact id, or null
149
public Long lookup( CacheIdentifier identifier )
151
switch( identifier.getType() )
154
return _contactsByName.get( identifier.getDetail() );
156
return _contactsByOrg.get( identifier.getDetail() );
158
return _contactsByNumber.get( identifier.getDetail() );
160
return _contactsByEmail.get( identifier.getDetail() );
166
* Remove any cache entry that is identified by the cache identifier.
167
* @param identifier the cache identifier
168
* @return the contact id of the contact that was removed, or null
170
public Long removeLookup( CacheIdentifier identifier )
172
switch( identifier.getType() )
175
return _contactsByName.remove( identifier.getDetail() );
177
return _contactsByOrg.remove( identifier.getDetail() );
179
return _contactsByNumber.remove( identifier.getDetail() );
181
return _contactsByEmail.remove( identifier.getDetail() );
187
* Add a lookup from a contact identifier to a contact id to the cache.
188
* @param identifier the cache identifier
189
* @param id teh contact id
191
public void addLookup( CacheIdentifier identifier, Long id )
193
switch( identifier.getType() )
196
_contactsByName.put( identifier.getDetail(), id );
199
_contactsByOrg.put( identifier.getDetail(), id );
202
_contactsByNumber.put( identifier.getDetail(), id );
205
_contactsByEmail.put( identifier.getDetail(), id );
211
* Remove any data that is associated with an contact id.
214
public void removeAssociatedData( Long id )
216
_contactNumbers.remove( id );
217
_contactEmails.remove( id );
218
_contactAddresses.remove( id );
219
_contactOrganisations.remove( id );
220
_contactNotes.remove( id );
223
public boolean hasAssociatedNumber( Long id, String number )
225
number = normalisePhoneNumber( number );
226
if( number == null ) return false;
228
HashSet< String > set = _contactNumbers.get( id );
229
return set != null && set.contains( number );
232
public void addAssociatedNumber( Long id, String number )
234
number = normalisePhoneNumber( number );
235
if( number == null ) return;
237
HashSet< String > set = _contactNumbers.get( id );
239
set = new HashSet< String >();
240
_contactNumbers.put( id, set );
245
public boolean hasAssociatedEmail( Long id, String email )
247
email = normaliseEmailAddress( email );
248
if( email == null ) return false;
250
HashSet< String > set = _contactEmails.get( id );
251
return set != null && set.contains( email );
254
public void addAssociatedEmail( Long id, String email )
256
email = normaliseEmailAddress( email );
257
if( email == null ) return;
259
HashSet< String > set = _contactEmails.get( id );
261
set = new HashSet< String >();
262
_contactEmails.put( id, set );
267
public boolean hasAssociatedAddress( Long id, String address )
269
address = normaliseAddress( address );
270
if( address == null ) return false;
272
HashSet< String > set = _contactAddresses.get( id );
273
return set != null && set.contains( address );
276
public void addAssociatedAddress( Long id, String address )
278
address = normaliseAddress( address );
279
if( address == null ) return;
281
HashSet< String > set = _contactAddresses.get( id );
283
set = new HashSet< String >();
284
_contactAddresses.put( id, set );
289
public boolean hasAssociatedOrganisation( Long id, String organisation )
291
organisation = normaliseOrganisation( organisation );
292
if( organisation == null ) return false;
294
HashSet< String > set = _contactOrganisations.get( id );
295
return set != null && set.contains( organisation );
298
public void addAssociatedOrganisation( Long id, String organisation )
300
organisation = normaliseOrganisation( organisation );
301
if( organisation == null ) return;
303
HashSet< String > set = _contactOrganisations.get( id );
305
set = new HashSet< String >();
306
_contactOrganisations.put( id, set );
308
set.add( organisation );
311
public boolean hasAssociatedNote( Long id, String note )
313
note = normaliseNote( note );
314
if( note == null ) return false;
316
HashSet< String > set = _contactNotes.get( id );
317
return set != null && set.contains( note );
320
public void addAssociatedNote( Long id, String note )
322
note = normaliseNote( note );
323
if( note == null ) return;
325
HashSet< String > set = _contactNotes.get( id );
327
set = new HashSet< String >();
328
_contactNotes.put( id, set );
333
static public String normaliseName( String name )
335
if( name == null ) return null;
337
return name.length() > 0? name : null;
340
static public String normalisePhoneNumber( String number )
342
if( number == null ) return null;
343
number = number.trim().replaceAll( "[-\\(\\) ]", "" );
344
return number.length() > 0? number : null;
347
static public String normaliseEmailAddress( String email )
349
if( email == null ) return null;
350
email = email.trim().toLowerCase( Locale.US );
351
return email.length() > 0? email : null;
354
static public String normaliseOrganisation( String organisation )
356
if( organisation == null ) return null;
357
organisation = organisation.trim();
358
return organisation.length() > 0? organisation : null;
361
static public String normaliseAddress( String address )
363
if( address == null ) return null;
364
address = address.trim();
365
return address.length() > 0? address : null;
368
static public String normaliseNote( String note )
370
if( note == null ) return null;
372
return note.length() > 0? note : null;
38
private HashMap< String, Long > _contacts;
40
private HashMap< Long, HashSet< String > > _contactNumbers;
41
private HashMap< Long, HashSet< String > > _contactEmails;
42
private HashMap< Long, HashSet< String > > _contactAddresses;
44
public boolean exists( String name )
46
return _contacts.containsKey( name );
49
public Long getId( String name )
51
return _contacts.get( name );
54
public void remove( String name )
56
_contacts.remove( name );
59
public void put( Long id, String name )
61
_contacts.put( name, id );
64
public boolean hasNumber( Long id, String number )
66
HashSet< String > cache = _contactNumbers.get( id );
67
return cache != null && cache.contains( number );
70
public void addNumber( Long id, String number )
72
HashSet< String > cache = _contactNumbers.get( id );
74
cache = new HashSet< String >();
75
_contactNumbers.put( id, cache );
80
public boolean hasEmail( Long id, String email )
82
HashSet< String > cache = _contactEmails.get( id );
83
return cache != null && cache.contains( email );
86
public void addEmail( Long id, String email )
88
HashSet< String > cache = _contactEmails.get( id );
90
cache = new HashSet< String >();
91
_contactEmails.put( id, cache );
96
public boolean hasAddress( Long id, String address )
98
HashSet< String > cache = _contactAddresses.get( id );
99
return cache != null && cache.contains( address );
102
public void addAddress( Long id, String address )
104
HashSet< String > cache = _contactAddresses.get( id );
105
if( cache == null ) {
106
cache = new HashSet< String >();
107
_contactAddresses.put( id, cache );
109
cache.add( address );
112
public void buildCache( Activity activity )
113
throws AbortImportException
118
// init contacts caches
119
_contacts = new HashMap< String, Long >();
120
_contactNumbers = new HashMap< Long, HashSet< String > >();
121
_contactEmails = new HashMap< Long, HashSet< String > >();
122
_contactAddresses = new HashMap< Long, HashSet< String > >();
124
// query and store map of contact names to ids
125
cols = new String[] { Contacts.People._ID, Contacts.People.NAME };
126
cur = activity.managedQuery( Contacts.People.CONTENT_URI,
127
cols, null, null, null);
128
if( cur.moveToFirst() ) {
129
int idCol = cur.getColumnIndex( Contacts.People._ID );
130
int nameCol = cur.getColumnIndex( Contacts.People.NAME );
132
_contacts.put( cur.getString( nameCol ), cur.getLong( idCol ) );
133
} while( cur.moveToNext() );
136
// query and store map of contact ids to sets of phone numbers
137
cols = new String[] { Contacts.Phones.PERSON_ID,
138
Contacts.Phones.NUMBER };
139
cur = activity.managedQuery( Contacts.Phones.CONTENT_URI,
140
cols, null, null, null);
141
if( cur.moveToFirst() ) {
142
int personIdCol = cur.getColumnIndex( Contacts.Phones.PERSON_ID );
143
int numberCol = cur.getColumnIndex( Contacts.Phones.NUMBER );
145
Long id = cur.getLong( personIdCol );
146
String number = Importer.sanitisePhoneNumber(
147
cur.getString( numberCol ) );
148
if( number != null ) {
149
HashSet< String > numbers = _contactNumbers.get( id );
150
if( numbers == null ) {
151
numbers = new HashSet< String >();
152
_contactNumbers.put( id, numbers );
154
numbers.add( number );
156
} while( cur.moveToNext() );
159
// query and store map of contact ids to sets of email addresses
160
cols = new String[] { Contacts.ContactMethods.PERSON_ID,
161
Contacts.ContactMethods.DATA };
162
cur = activity.managedQuery( Contacts.ContactMethods.CONTENT_URI,
163
cols, Contacts.ContactMethods.KIND + " = ?",
164
new String[] { "" + Contacts.KIND_EMAIL }, null );
165
if( cur.moveToFirst() ) {
166
int personIdCol = cur.getColumnIndex(
167
Contacts.ContactMethods.PERSON_ID );
168
int addressCol = cur.getColumnIndex(
169
Contacts.ContactMethods.DATA );
171
Long id = cur.getLong( personIdCol );
172
String address = Importer.sanitiseEmailAddress(
173
cur.getString( addressCol ) );
174
if( address != null ) {
175
HashSet< String > addresses = _contactEmails.get( id );
176
if( addresses == null ) {
177
addresses = new HashSet< String >();
178
_contactEmails.put( id, addresses );
180
addresses.add( address );
182
} while( cur.moveToNext() );
185
// query and store map of contact ids to sets of postal addresses
186
cols = new String[] { Contacts.ContactMethods.PERSON_ID,
187
Contacts.ContactMethods.DATA };
188
cur = activity.managedQuery( Contacts.ContactMethods.CONTENT_URI,
189
cols, Contacts.ContactMethods.KIND + " = ?",
190
new String[] { "" + Contacts.KIND_POSTAL }, null );
191
if( cur.moveToFirst() ) {
192
int personIdCol = cur.getColumnIndex(
193
Contacts.ContactMethods.PERSON_ID );
194
int addressCol = cur.getColumnIndex(
195
Contacts.ContactMethods.DATA );
197
Long id = cur.getLong( personIdCol );
198
String address = cur.getString( addressCol );
199
if( address != null ) {
200
HashSet< String > addresses = _contactAddresses.get( id );
201
if( addresses == null ) {
202
addresses = new HashSet< String >();
203
_contactAddresses.put( id, addresses );
205
addresses.add( address );
207
} while( cur.moveToNext() );