Home | History | Annotate | Download | only in latin
      1 /*
      2  * Copyright (C) 2014 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.inputmethod.latin;
     18 
     19 import android.Manifest;
     20 import android.content.ContentResolver;
     21 import android.content.Context;
     22 import android.database.ContentObserver;
     23 import android.os.SystemClock;
     24 import android.provider.ContactsContract.Contacts;
     25 import android.util.Log;
     26 
     27 import com.android.inputmethod.latin.ContactsManager.ContactsChangedListener;
     28 import com.android.inputmethod.latin.define.DebugFlags;
     29 import com.android.inputmethod.latin.permissions.PermissionsUtil;
     30 import com.android.inputmethod.latin.utils.ExecutorUtils;
     31 
     32 import java.util.ArrayList;
     33 import java.util.concurrent.atomic.AtomicBoolean;
     34 
     35 /**
     36  * A content observer that listens to updates to content provider {@link Contacts#CONTENT_URI}.
     37  */
     38 public class ContactsContentObserver implements Runnable {
     39     private static final String TAG = "ContactsContentObserver";
     40 
     41     private final Context mContext;
     42     private final ContactsManager mManager;
     43     private final AtomicBoolean mRunning = new AtomicBoolean(false);
     44 
     45     private ContentObserver mContentObserver;
     46     private ContactsChangedListener mContactsChangedListener;
     47 
     48     public ContactsContentObserver(final ContactsManager manager, final Context context) {
     49         mManager = manager;
     50         mContext = context;
     51     }
     52 
     53     public void registerObserver(final ContactsChangedListener listener) {
     54         if (!PermissionsUtil.checkAllPermissionsGranted(
     55                 mContext, Manifest.permission.READ_CONTACTS)) {
     56             Log.i(TAG, "No permission to read contacts. Not registering the observer.");
     57             // do nothing if we do not have the permission to read contacts.
     58             return;
     59         }
     60 
     61         if (DebugFlags.DEBUG_ENABLED) {
     62             Log.d(TAG, "registerObserver()");
     63         }
     64         mContactsChangedListener = listener;
     65         mContentObserver = new ContentObserver(null /* handler */) {
     66             @Override
     67             public void onChange(boolean self) {
     68                 ExecutorUtils.getBackgroundExecutor(ExecutorUtils.KEYBOARD)
     69                         .execute(ContactsContentObserver.this);
     70             }
     71         };
     72         final ContentResolver contentResolver = mContext.getContentResolver();
     73         contentResolver.registerContentObserver(Contacts.CONTENT_URI, true, mContentObserver);
     74     }
     75 
     76     @Override
     77     public void run() {
     78         if (!PermissionsUtil.checkAllPermissionsGranted(
     79                 mContext, Manifest.permission.READ_CONTACTS)) {
     80             Log.i(TAG, "No permission to read contacts. Not updating the contacts.");
     81             unregister();
     82             return;
     83         }
     84 
     85         if (!mRunning.compareAndSet(false /* expect */, true /* update */)) {
     86             if (DebugFlags.DEBUG_ENABLED) {
     87                 Log.d(TAG, "run() : Already running. Don't waste time checking again.");
     88             }
     89             return;
     90         }
     91         if (haveContentsChanged()) {
     92             if (DebugFlags.DEBUG_ENABLED) {
     93                 Log.d(TAG, "run() : Contacts have changed. Notifying listeners.");
     94             }
     95             mContactsChangedListener.onContactsChange();
     96         }
     97         mRunning.set(false);
     98     }
     99 
    100     boolean haveContentsChanged() {
    101         if (!PermissionsUtil.checkAllPermissionsGranted(
    102                 mContext, Manifest.permission.READ_CONTACTS)) {
    103             Log.i(TAG, "No permission to read contacts. Marking contacts as not changed.");
    104             return false;
    105         }
    106 
    107         final long startTime = SystemClock.uptimeMillis();
    108         final int contactCount = mManager.getContactCount();
    109         if (contactCount > ContactsDictionaryConstants.MAX_CONTACTS_PROVIDER_QUERY_LIMIT) {
    110             // If there are too many contacts then return false. In this rare case it is impossible
    111             // to include all of them anyways and the cost of rebuilding the dictionary is too high.
    112             // TODO: Sort and check only the most recent contacts?
    113             return false;
    114         }
    115         if (contactCount != mManager.getContactCountAtLastRebuild()) {
    116             if (DebugFlags.DEBUG_ENABLED) {
    117                 Log.d(TAG, "haveContentsChanged() : Count changed from "
    118                         + mManager.getContactCountAtLastRebuild() + " to " + contactCount);
    119             }
    120             return true;
    121         }
    122         final ArrayList<String> names = mManager.getValidNames(Contacts.CONTENT_URI);
    123         if (names.hashCode() != mManager.getHashCodeAtLastRebuild()) {
    124             return true;
    125         }
    126         if (DebugFlags.DEBUG_ENABLED) {
    127             Log.d(TAG, "haveContentsChanged() : No change detected in "
    128                     + (SystemClock.uptimeMillis() - startTime) + " ms)");
    129         }
    130         return false;
    131     }
    132 
    133     public void unregister() {
    134         mContext.getContentResolver().unregisterContentObserver(mContentObserver);
    135     }
    136 }
    137