/android/import-contacts

To get this branch, use:
bzr branch http://bzr.ed.am/android/import-contacts

« back to all changes in this revision

Viewing changes to src/am/ed/importcontacts/ContactsBackend.java

  • Committer: edam
  • Date: 2012-12-18 11:58:46 UTC
  • Revision ID: tim@ed.am-20121218115846-xdwnw3zzcedbb3rp
bump version to 1.3 (we're skipping v1.2, for f-droid)

Show diffs side-by-side

added added

removed removed

23
23
 
24
24
package am.ed.importcontacts;
25
25
 
26
 
import java.util.HashMap;
27
26
import java.util.HashSet;
28
 
import java.util.Iterator;
29
27
 
30
28
import am.ed.importcontacts.ContactsCache.CacheIdentifier;
31
29
import am.ed.importcontacts.Importer.ContactData;
36
34
import android.net.Uri;
37
35
import android.provider.Contacts;
38
36
 
39
 
@SuppressWarnings( "deprecation" )
 
37
 
40
38
public class ContactsBackend implements Backend
41
39
{
42
40
        private Activity _activity = null;
52
50
                Cursor cur;
53
51
 
54
52
                // set of contact ids that we have not yet added
55
 
                HashSet< Long > unadded_ids = new HashSet< Long >();
56
 
 
57
 
                // notes
58
 
                HashMap< Long, String > notes = new HashMap< Long, String >();
 
53
                HashSet< Long > unadded = new HashSet< Long >();
59
54
 
60
55
                // get all contacts
61
56
                cur = _activity.managedQuery( Contacts.People.CONTENT_URI,
62
57
                        new String[] {
63
58
                                Contacts.People._ID,
64
59
                                Contacts.People.NAME,
65
 
                                Contacts.People.NOTES,
66
60
                        }, null, null, null );
67
61
                while( cur.moveToNext() ) {
68
62
                        Long id = cur.getLong(
69
63
                                cur.getColumnIndex( Contacts.People._ID ) );
70
 
                        String name = cur.getString(
71
 
                                        cur.getColumnIndex( Contacts.People.NAME ) );
72
 
                        String note = cur.getString(
73
 
                                        cur.getColumnIndex( Contacts.People.NOTES ) );
74
 
 
75
 
                        // if we can, add a lookup for the contact id by name
76
 
                        CacheIdentifier cache_identifier = CacheIdentifier.factory(
77
 
                                CacheIdentifier.Type.NAME, name );
78
 
                        if( cache_identifier != null ) {
79
 
                                cache.addLookup( cache_identifier, id );
80
 
 
81
 
                                // add any associated notes (this would get done at the end but,
82
 
                                // since it is most common that contacts are identified by name,
83
 
                                // it is worth doing a special case here
84
 
                                cache.addAssociatedNote( id, note );
85
 
                        }
86
 
                        else
 
64
                        String name =
 
65
                                ContactsCache.normaliseName( cur.getString(
 
66
                                        cur.getColumnIndex( Contacts.People.NAME ) ) );
 
67
                        if( name != null )
87
68
                        {
88
 
                                // record that a lookup for this contact's id still needs to be
89
 
                                // added by some other means
90
 
                                unadded_ids.add( id );
 
69
                                // if we can, add a lookup for the contact id by name
 
70
                                if( name.length() > 0 ) {
 
71
                                        cache.addLookup( new CacheIdentifier(
 
72
                                                CacheIdentifier.Type.NAME, name ), id );
 
73
                                        continue;
 
74
                                }
 
75
                        }
91
76
 
92
 
                                // store this contact's notes, so that they can be added to the
93
 
                                // cache at the end, after this contact has been added (by
94
 
                                // whatever identifying means)
95
 
                                if( note != null && note.length() > 0 )
96
 
                                        notes.put( id, note );
97
 
                        }
 
77
                        // record that a lookup for this contact's id still needs to be
 
78
                        // added by some other means
 
79
                        unadded.add( id );
98
80
                }
99
81
 
100
82
                // get contact organisations, primary ones first
106
88
                while( cur.moveToNext() ) {
107
89
                        Long id = cur.getLong( cur.getColumnIndex(
108
90
                                Contacts.Organizations.PERSON_ID ) );
109
 
                        String organisation = cur.getString(
110
 
                                cur.getColumnIndex( Contacts.Organizations.COMPANY ) );
111
 
 
112
 
                        // if this is an organisation name for a contact for whom we have
113
 
                        // not added a lookup, add a lookup for the contact id by
114
 
                        // organisation
115
 
                        if( unadded_ids.contains( id ) ) {
116
 
                                CacheIdentifier cache_identifier = CacheIdentifier.factory(
117
 
                                        CacheIdentifier.Type.ORGANISATION, organisation );
118
 
                                if( cache_identifier != null ) {
119
 
                                        cache.addLookup( cache_identifier, id );
120
 
                                        unadded_ids.remove( id );
 
91
                        String organisation =
 
92
                                ContactsCache.normaliseOrganisation( cur.getString(
 
93
                                        cur.getColumnIndex( Contacts.Organizations.COMPANY ) ) );
 
94
                        if( organisation != null )
 
95
                        {
 
96
                                // if this is an organisation name for a contact for whom we
 
97
                                // have not added a lookup, add a lookup for the contact id
 
98
                                // by organisation
 
99
                                if( unadded.contains( id ) ) {
 
100
                                        cache.addLookup( new CacheIdentifier(
 
101
                                                CacheIdentifier.Type.ORGANISATION, organisation ), id );
 
102
                                        unadded.remove( id );
121
103
                                }
 
104
 
 
105
                                // add associated data
 
106
                                cache.addAssociatedOrganisation( id, organisation );
122
107
                        }
123
 
 
124
 
                        // add associated data
125
 
                        cache.addAssociatedOrganisation( id, organisation );
126
108
                }
127
109
 
128
110
                // get all phone numbers, primary ones first
134
116
                while( cur.moveToNext() ) {
135
117
                        Long id = cur.getLong(
136
118
                                cur.getColumnIndex( Contacts.Phones.PERSON_ID ) );
137
 
                        String number = cur.getString(
138
 
                                cur.getColumnIndex( Contacts.Phones.NUMBER ) );
139
 
 
140
 
                        // if this is a number for a contact for whom we have not
141
 
                        // added a lookup, add a lookup for the contact id by phone
142
 
                        // number
143
 
                        if( unadded_ids.contains( id ) ) {
144
 
                                CacheIdentifier cache_identifier = CacheIdentifier.factory(
145
 
                                        CacheIdentifier.Type.PRIMARY_NUMBER, number );
146
 
                                if( cache_identifier != null ) {
147
 
                                        cache.addLookup( cache_identifier, id );
148
 
                                        unadded_ids.remove( id );
 
119
                        String number =
 
120
                                ContactsCache.normalisePhoneNumber( cur.getString(
 
121
                                        cur.getColumnIndex( Contacts.Phones.NUMBER ) ) );
 
122
                        if( number != null )
 
123
                        {
 
124
                                // if this is a number for a contact for whom we have not
 
125
                                // added a lookup, add a lookup for the contact id by phone
 
126
                                // number
 
127
                                if( unadded.contains( id ) ) {
 
128
                                        cache.addLookup( new CacheIdentifier(
 
129
                                                CacheIdentifier.Type.PRIMARY_NUMBER, number ), id );
 
130
                                        unadded.remove( id );
149
131
                                }
 
132
 
 
133
                                // add associated data
 
134
                                cache.addAssociatedNumber( id, number );
150
135
                        }
151
 
 
152
 
                        // add associated data
153
 
                        cache.addAssociatedNumber( id, number );
154
136
                }
155
137
 
156
138
                // now get all email addresses, primary ones first, and postal addresses
159
141
                                Contacts.ContactMethods.PERSON_ID,
160
142
                                Contacts.ContactMethods.DATA,
161
143
                                Contacts.ContactMethods.KIND,
162
 
                        }, Contacts.ContactMethods.KIND + " IN( ?, ? )",
163
 
                        new String[] {
 
144
                        }, Contacts.ContactMethods.KIND + " IN( ?, ? )", new String[] {
164
145
                                "" + Contacts.KIND_EMAIL,
165
146
                                "" + Contacts.KIND_POSTAL,
166
147
                        }, Contacts.ContactMethods.ISPRIMARY + " DESC" );
171
152
                                cur.getColumnIndex( Contacts.ContactMethods.KIND ) );
172
153
                        if( kind == Contacts.KIND_EMAIL )
173
154
                        {
174
 
                                String email = cur.getString(
175
 
                                        cur.getColumnIndex( Contacts.ContactMethods.DATA ) );
176
 
 
177
 
                                // if this is an email address for a contact for whom we have
178
 
                                // not added a lookup, add a lookup for the contact id by email
179
 
                                // address
180
 
                                if( unadded_ids.contains( id ) ) {
181
 
                                        CacheIdentifier cache_identifier = CacheIdentifier.factory(
182
 
                                                CacheIdentifier.Type.PRIMARY_EMAIL, email );
183
 
                                        if( cache_identifier != null ) {
184
 
                                                cache.addLookup( cache_identifier, id );
185
 
                                                unadded_ids.remove( id );
 
155
                                String email =
 
156
                                        ContactsCache.normaliseEmailAddress( cur.getString(
 
157
                                                cur.getColumnIndex( Contacts.ContactMethods.DATA ) ) );
 
158
                                if( email != null )
 
159
                                {
 
160
                                        // if this is an email address for a contact for whom we
 
161
                                        // have not added a lookup, add a lookup for the contact
 
162
                                        // id by email address
 
163
                                        if( unadded.contains( id ) ) {
 
164
                                                cache.addLookup( new CacheIdentifier(
 
165
                                                        CacheIdentifier.Type.PRIMARY_EMAIL, email ), id );
 
166
                                                unadded.remove( id );
186
167
                                        }
 
168
 
 
169
                                        // add associated data
 
170
                                        cache.addAssociatedEmail( id, email );
187
171
                                }
188
 
 
189
 
                                // add associated data
190
 
                                cache.addAssociatedEmail( id, email );
191
172
                        }
192
173
                        else if( kind == Contacts.KIND_POSTAL )
193
174
                        {
194
 
                                String address = cur.getString(
195
 
                                        cur.getColumnIndex( Contacts.ContactMethods.DATA ) );
196
 
 
197
 
                                // add associated data
198
 
                                cache.addAssociatedAddress( id, address );
 
175
                                String address =
 
176
                                        ContactsCache.normaliseAddress( cur.getString(
 
177
                                                cur.getColumnIndex( Contacts.ContactMethods.DATA ) ) );
 
178
                                if( address != null )
 
179
                                {
 
180
                                        // add associated data
 
181
                                        cache.addAssociatedAddress( id, address );
 
182
                                }
199
183
                        }
200
184
                }
201
 
 
202
 
                // finally, add the notes that we stored earlier (we have to add these
203
 
                // at the end because we can't be sure which piece of contact data will
204
 
                // cause the contact to be added to the cache
205
 
                Iterator< Long > i = notes.keySet().iterator();
206
 
                while( i.hasNext() ) {
207
 
                        Long id = i.next();
208
 
                        cache.addAssociatedNote( id, notes.get( id ) );
209
 
                }
210
185
        }
211
186
 
212
187
        @Override
218
193
        }
219
194
 
220
195
        @Override
221
 
        public Long addContact( String name ) throws ContactCreationException
 
196
        public Long addContact( String name )
222
197
        {
223
198
                ContentValues values = new ContentValues();
224
 
                if( name != null )
225
 
                        values.put( Contacts.People.NAME, name );
 
199
                values.put( Contacts.People.NAME, name );
226
200
                Uri contact_uri = _activity.getContentResolver().insert(
227
201
                        Contacts.People.CONTENT_URI, values );
228
202
                Long id = ContentUris.parseId( contact_uri );
229
 
                if( id == 0 )
230
 
                        throw new ContactCreationException();
231
203
 
232
204
                // try to add them to the "My Contacts" group
233
 
                try {
234
 
                        Contacts.People.addToMyContactsGroup(
235
 
                                _activity.getContentResolver(), id );
236
 
                }
237
 
                catch( IllegalStateException e ) {
238
 
                        // ignore any failure
239
 
                }
240
 
 
241
 
                return id;
242
 
        }
243
 
 
244
 
        private int convertTypeToBackendType( Class< ? > cls, int type )
245
 
                throws ContactCreationException
246
 
        {
247
 
                if( cls == Contacts.Phones.class )
248
 
                {
249
 
                        switch( type )
250
 
                        {
251
 
                        case ContactData.TYPE_HOME:
252
 
                                return Contacts.Phones.TYPE_HOME;
253
 
                        case ContactData.TYPE_WORK:
254
 
                                return Contacts.Phones.TYPE_WORK;
255
 
                        case ContactData.TYPE_MOBILE:
256
 
                                return Contacts.Phones.TYPE_MOBILE;
257
 
                        case ContactData.TYPE_FAX_HOME:
258
 
                                return Contacts.Phones.TYPE_FAX_HOME;
259
 
                        case ContactData.TYPE_FAX_WORK:
260
 
                                return Contacts.Phones.TYPE_FAX_WORK;
261
 
                        case ContactData.TYPE_PAGER:
262
 
                                return Contacts.Phones.TYPE_PAGER;
263
 
                        }
264
 
                }
265
 
                else if( cls == Contacts.ContactMethods.class )
266
 
                {
267
 
                        switch( type )
268
 
                        {
269
 
                        case ContactData.TYPE_HOME:
270
 
                                return Contacts.ContactMethods.TYPE_HOME;
271
 
                        case ContactData.TYPE_WORK:
272
 
                                return Contacts.ContactMethods.TYPE_WORK;
273
 
                        }
274
 
                }
275
 
 
276
 
                // still here?
277
 
                throw new ContactCreationException();
 
205
                if( id != null && id > 0 ) {
 
206
                        try {
 
207
                                Contacts.People.addToMyContactsGroup(
 
208
                                        _activity.getContentResolver(), id );
 
209
                        }
 
210
                        catch( IllegalStateException e ) {
 
211
                                // ignore any failure
 
212
                        }
 
213
                }
 
214
 
 
215
                return id == null? 0 : id;
278
216
        }
279
217
 
280
218
        @Override
281
219
        public void addContactPhone( Long id, String number,
282
 
                ContactData.PreferredDetail data ) throws ContactCreationException
 
220
                ContactData.PreferredDetail data )
283
221
        {
284
222
                Uri contact_phones_uri = Uri.withAppendedPath(
285
223
                        ContentUris.withAppendedId( Contacts.People.CONTENT_URI, id ),
286
224
                        Contacts.People.Phones.CONTENT_DIRECTORY );
287
225
 
288
226
                ContentValues values = new ContentValues();
289
 
                values.put( Contacts.Phones.TYPE,
290
 
                        convertTypeToBackendType( Contacts.Phones.class,
291
 
                                data.getType() ) );
 
227
                values.put( Contacts.Phones.TYPE, data.getType() );
292
228
                values.put( Contacts.Phones.NUMBER, number );
293
229
                if( data.isPreferred() )
294
230
                        values.put( Contacts.Phones.ISPRIMARY, 1 );
298
234
 
299
235
        @Override
300
236
        public void addContactEmail( Long id, String email,
301
 
                ContactData.PreferredDetail data ) throws ContactCreationException
 
237
                ContactData.PreferredDetail data )
302
238
        {
303
239
                Uri contact_contact_methods_uri = Uri.withAppendedPath(
304
240
                        ContentUris.withAppendedId( Contacts.People.CONTENT_URI, id ),
307
243
                ContentValues values = new ContentValues();
308
244
                values.put( Contacts.ContactMethods.KIND, Contacts.KIND_EMAIL );
309
245
                values.put( Contacts.ContactMethods.DATA, email );
310
 
                values.put( Contacts.ContactMethods.TYPE,
311
 
                        convertTypeToBackendType( Contacts.ContactMethods.class,
312
 
                                data.getType() ) );
 
246
                values.put( Contacts.ContactMethods.TYPE, data.getType() );
313
247
                if( data.isPreferred() )
314
248
                        values.put( Contacts.ContactMethods.ISPRIMARY, 1 );
315
249
 
319
253
 
320
254
        @Override
321
255
        public void addContactAddresses( Long id, String address,
322
 
                ContactData.TypeDetail data ) throws ContactCreationException
 
256
                ContactData.TypeDetail data )
323
257
        {
324
258
                Uri contact_contact_methods_uri = Uri.withAppendedPath(
325
259
                        ContentUris.withAppendedId( Contacts.People.CONTENT_URI, id ),
328
262
                ContentValues values = new ContentValues();
329
263
                values.put( Contacts.ContactMethods.KIND, Contacts.KIND_POSTAL );
330
264
                values.put( Contacts.ContactMethods.DATA, address );
331
 
                values.put( Contacts.ContactMethods.TYPE,
332
 
                        convertTypeToBackendType( Contacts.ContactMethods.class,
333
 
                                data.getType() ) );
 
265
                values.put( Contacts.ContactMethods.TYPE, data.getType() );
334
266
 
335
267
                _activity.getContentResolver().insert( contact_contact_methods_uri,
336
268
                        values );
338
270
 
339
271
        @Override
340
272
        public void addContactOrganisation( Long id, String organisation,
341
 
                ContactData.ExtraDetail data ) throws ContactCreationException
 
273
                ContactData.ExtraDetail data )
342
274
        {
343
275
                ContentValues values = new ContentValues();
344
276
                values.put( Contacts.Organizations.PERSON_ID, id );
352
284
                        Contacts.Organizations.CONTENT_URI, values );
353
285
        }
354
286
 
355
 
        @Override
356
 
        public void addContactNote( Long id, String note )
357
 
                throws ContactCreationException
358
 
        {
359
 
                ContentValues values = new ContentValues();
360
 
                values.put( Contacts.People.NOTES, note );
361
 
                _activity.getContentResolver().update(
362
 
                        ContentUris.withAppendedId( Contacts.People.CONTENT_URI, id ),
363
 
                        values, null, null );
364
 
        }
365
287
 
366
288
}