How do I make my Android ContentObserver for ContactsContract detect a added, updated or deleted contact?
Asked Answered
R

4

32

I am able to get a generic notification "that there was a change to the contacts DB", but I want to know the specific record that was inserted, updated, or deleted. Following is the code that gets registered and gets the onChange notification. Unfortunately, it is not specific which makes my processing exhaustive and inefficient.

Here is the code stub:

            if ((mNativeContactsObserver == null) && (mHandler == null)) {
            mHandler = new Handler(this.getMainLooper()) {
            };
            mNativeContactsObserver = new ContentObserver(mHandler) {
                @Override
                public void onChange(boolean selfChange) {
                    super.onChange(selfChange);

                    Bundle data = null;
                    Message message = mHandler.obtainMessage();
                    if (message != null) {
                        data = message.getData();
                        if (data != null) {
                            Logs.d(TAG, "Message = [" + message.toString() + "] data=[" + data.toString() + "]");
                            Logs.d(TAG, "Contents = [" + message.describeContents() + "]");
                        }
                    }

                    if (!selfChange) {
                        final Account accountListen = MySyncAdapter.lookupAccount(TAG, getApplicationContext(), getUserProfile().getAccountId(), AUTHORITY_MY_SYNC);
                        Logs.d(TAG, "onChange!? account: " + accountListen.name);
                        if (!ContentResolver.isSyncPending(account, ContactsContract.AUTHORITY) 
                                && (!ContentResolver.isSyncActive(account, ContactsContract.AUTHORITY))) {
                            Bundle extras = new Bundle();
                            extras.putInt(MySyncAdapter.EXTRA_SYNC_TYPE, MySyncAdapter.REQUEST_SYNC_NATIVE_CHANGED);
                            ContentResolver.requestSync(accountListen, ContactsContract.AUTHORITY, extras);
                        } else {
                            Logs.w(TAG, "There is a pending sync.  This request is ignored.");
                        }
                    }
                }
            };
        }
        Uri uriContactsListen = ContactsContract.Contacts.CONTENT_URI.buildUpon().appendEncodedPath("#").build();
        Logs.i(TAG, "Register listening for native contacts changes. [" + uriContactsListen + "]");
        cr.registerContentObserver(uriContactsListen, true, mNativeContactsObserver);
Ralline answered 13/5, 2011 at 19:39 Comment(4)
Did you find out how to detect ContactsContract addition or deletion or when it is updated explicitly?Even i want my ContentObserver to be notified only when there is delete of record. Not every timeBoodle
Unfortunately not, however, I have not pursued since Eclair.Ralline
hi, have you got the solution for it? i want the same result in my project.Eagan
@BhoomiZalavadiya unfortunately, no. i just wrote if off as a limitation and never looked back.Ralline
H
5

Finally at least i am able to detect is this Update/Delete/Insert look at my OnChange method as below

 @Override
       public void onChange(boolean selfChange) {
           super.onChange(selfChange);
           Log.e("ContactChangeObserver", "onChange");

           // 0 Update , 1 Delete , 2 Added
 // Get count from phone contacts
           final int currentCount = contactDBOperaion.getContactsCountFromPhone();

 // Get count from your sqlite database
           int mContactCount= DbContacts.getInstance().getContactsCount();

           if (currentCount < mContactCount) {
               // DELETE HAPPEN.
               Log.e("Status", "Deletion");
               contactDBOperaion.SyncContacts(1);
           } else if (currentCount == mContactCount) {
               // UPDATE HAPPEN.
               contactDBOperaion.SyncContacts(0);
           } else {
               // INSERT HAPPEN.
               Log.e("Status", "Insertion");
               contactDBOperaion.SyncContacts(2);
           }
       }

When Update occurred i update all contacts

When Insert occurred i am inserting newly added row , after checking isExist in existing database

When Delete occurred not found solution yet

Heinrick answered 12/8, 2015 at 16:7 Comment(0)
P
3

I have this code in my Application base class.

private ContentObserver contactObserver = new ContactObserver();

private class ContactObserver extends ContentObserver {

    public ContactObserver() {
        super(null);
    }

    @Override
    public void onChange(boolean selfChange) {
        super.onChange(selfChange);

        // Since onChange do not sent which user have been changed, you 
        // have to figure out how to match it with your data.
        // Note: Contact is  one of my classes.
        for (Contact contact : getContacts()) {
            if (!contact.isLinked())
                continue;

            String selection = ContactsContract.Data._ID + " = ?";
            String[] selectionArgs = new String[] { contact.getSystemId() };
            String[] projection = new String[] { ContactsContract.Data.DISPLAY_NAME };
            Cursor cursor = getContentResolver().query(
                    ContactsContract.Contacts.CONTENT_URI, projection,
                    selection, selectionArgs, null);

            if (!cursor.moveToFirst())
                return;

            String name = cursor.getString(0);

            if (contact.getUsername().equalsIgnoreCase(name))
                continue;

            contact.setUserName(name);

        }
    }
}

Regarding about what you can put in projection check here

Unfortunately, it is not specific which makes my processing exhaustive and inefficient.

Would be great if into next Android releases also will send the contact id which had been just changed.

Hope this helps

Perdita answered 21/5, 2011 at 21:37 Comment(2)
Can you help me i want to do when any new contact is added in the phonebook.then i would be notified and that new contact is send to the webservice on server... @Cuddle
To get more help from the whole community, please create a new question about your blockers, issues, concern, etc.Perdita
W
1

First of all, you have to register your contentObserver to receive change notification.

Do this by calling :

registerContentObserver();

Here's the specs : registerContetObserver

After than you'll want to notify all the listeners when a modification happens :

contentResolver.notifyChange();

Here's the specs for that one : notifyChange

Hope it helps ;)

Cheers !

Weatherby answered 18/5, 2011 at 17:48 Comment(6)
oops! I didn't mention that I am doing the registration and getting the notification (i.e., from someone using the Android Address Book - Contacts App). What I am trying to get is the record of the contact so I can capture a reference in my data structure -- instead, what I have to do is search the entire address book for any changes. My thinking is that I may need to provide a more specific URI to observe, or there is a way to extract a key from the notification. Oooorrrr ... something completely different :)Ralline
Ok. I don't have your full code structure here so it's a bit hard to figure all of it...But, can't you make each record register to an observer so that you get notified directly by the record when something has changed ;)Weatherby
Actually all of the code is in the description. As for registering for each record is not practical. There could be 500+ aggregated contacts which would be double (or more) raw contacts to register for. That is why I was hoping that the URI can be specified to include the sub-query. For example, you can see that I tried ContactsContract.Contacts.CONTENT_URI.buildUpon().appendEncodedPath("#").build(); -- it did not have the effect I was hoping.Ralline
Ok I see....Then maybe you could override the URI object....defining your own custom objects and embed the needed sub-query infos yourself... ;)Weatherby
how does one override a URI? i would have to subclass the content provider of the URI.Ralline
Yup ! Extends the ContentProvider class and define your own method implementation. Lookup that following example ;) thinkandroid.wordpress.com/2010/01/13/…Weatherby
S
-2

Use Loaders to query and display the list of contacts on your android device on your emulator. They automatically detect changes in the underlying data and reflect that on your UI.

Stuckup answered 24/5, 2013 at 6:18 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.