1 /* 2 * Copyright (C) 2016 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 package com.android.contacts.util.concurrent; 17 18 import android.content.BroadcastReceiver; 19 import android.content.Context; 20 import android.content.Intent; 21 import android.content.IntentFilter; 22 import android.content.Loader; 23 import androidx.localbroadcastmanager.content.LocalBroadcastManager; 24 import android.util.Log; 25 26 import com.google.common.util.concurrent.FutureCallback; 27 import com.google.common.util.concurrent.Futures; 28 import com.google.common.util.concurrent.ListenableFuture; 29 30 import java.util.concurrent.CancellationException; 31 import java.util.concurrent.Executor; 32 33 /** 34 * Wraps a ListenableFuture for integration with {@link android.app.LoaderManager} 35 * 36 * <p>Using a loader ensures that the result is delivered while the receiving component (activity 37 * or fragment) is resumed and also prevents leaking references these components 38 * </p> 39 */ 40 public abstract class ListenableFutureLoader<D> extends Loader<D> { 41 private static final String TAG = "FutureLoader"; 42 43 private final IntentFilter mReloadFilter; 44 private final Executor mUiExecutor; 45 private final LocalBroadcastManager mLocalBroadcastManager; 46 47 private ListenableFuture<D> mFuture; 48 private D mLoadedData; 49 50 private BroadcastReceiver mReceiver; 51 52 /** 53 * Stores away the application context associated with context. 54 * Since Loaders can be used across multiple activities it's dangerous to 55 * store the context directly; always use {@link #getContext()} to retrieve 56 * the Loader's Context, don't use the constructor argument directly. 57 * The Context returned by {@link #getContext} is safe to use across 58 * Activity instances. 59 * 60 * @param context used to retrieve the application context. 61 */ ListenableFutureLoader(Context context)62 public ListenableFutureLoader(Context context) { 63 this(context, null); 64 } 65 ListenableFutureLoader(Context context, IntentFilter reloadBroadcastFilter)66 public ListenableFutureLoader(Context context, IntentFilter reloadBroadcastFilter) { 67 super(context); 68 mUiExecutor = ContactsExecutors.newUiThreadExecutor(); 69 mReloadFilter = reloadBroadcastFilter; 70 mLocalBroadcastManager = LocalBroadcastManager.getInstance(context); 71 } 72 73 @Override onStartLoading()74 protected void onStartLoading() { 75 if (mReloadFilter != null && mReceiver == null) { 76 mReceiver = new ForceLoadReceiver(); 77 mLocalBroadcastManager.registerReceiver(mReceiver, mReloadFilter); 78 } 79 80 if (mLoadedData != null) { 81 deliverResult(mLoadedData); 82 } 83 if (mFuture == null) { 84 takeContentChanged(); 85 forceLoad(); 86 } else if (takeContentChanged()) { 87 forceLoad(); 88 } 89 } 90 91 @Override onForceLoad()92 protected void onForceLoad() { 93 mFuture = loadData(); 94 Futures.addCallback(mFuture, new FutureCallback<D>() { 95 @Override 96 public void onSuccess(D result) { 97 if (mLoadedData == null || !isSameData(mLoadedData, result)) { 98 deliverResult(result); 99 } 100 mLoadedData = result; 101 commitContentChanged(); 102 } 103 104 @Override 105 public void onFailure(Throwable t) { 106 if (t instanceof CancellationException) { 107 Log.i(TAG, "Loading cancelled", t); 108 rollbackContentChanged(); 109 } else { 110 Log.e(TAG, "Loading failed", t); 111 } 112 } 113 }, mUiExecutor); 114 } 115 116 @Override onStopLoading()117 protected void onStopLoading() { 118 if (mFuture != null) { 119 mFuture.cancel(false); 120 mFuture = null; 121 } 122 } 123 124 @Override onReset()125 protected void onReset() { 126 mFuture = null; 127 mLoadedData = null; 128 if (mReceiver != null) { 129 mLocalBroadcastManager.unregisterReceiver(mReceiver); 130 } 131 } 132 loadData()133 protected abstract ListenableFuture<D> loadData(); 134 135 /** 136 * Returns whether the newly loaded data is the same as the cached value 137 * 138 * <p>This allows subclasses to suppress delivering results when the data hasn't 139 * actually changed. By default it will always return false. 140 * </p> 141 */ isSameData(D previousData, D newData)142 protected boolean isSameData(D previousData, D newData) { 143 return false; 144 } 145 getLoadedData()146 public final D getLoadedData() { 147 return mLoadedData; 148 } 149 150 public class ForceLoadReceiver extends BroadcastReceiver { 151 @Override onReceive(Context context, Intent intent)152 public void onReceive(Context context, Intent intent) { 153 onContentChanged(); 154 } 155 } 156 } 157