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.server.telecom;
18 
19 import android.app.Notification;
20 import android.content.Context;
21 import android.graphics.Bitmap;
22 import android.graphics.drawable.BitmapDrawable;
23 import android.graphics.drawable.Drawable;
24 import android.telecom.Log;
25 import android.net.Uri;
26 import android.os.Handler;
27 import android.os.HandlerThread;
28 import android.os.Looper;
29 import android.os.Message;
30 
31 // TODO: Needed for move to system service: import com.android.internal.R;
32 
33 import java.io.FileNotFoundException;
34 import java.io.IOException;
35 import java.io.InputStream;
36 
37 /**
38  * Helper class for loading contacts photo asynchronously.
39  */
40 public class ContactsAsyncHelper {
41     private static final String LOG_TAG = ContactsAsyncHelper.class.getSimpleName();
42 
43     public static class Factory {
44         public ContactsAsyncHelper create(ContentResolverAdapter adapter) {
45             return new ContactsAsyncHelper(adapter);
46         }
47     }
48 
49     /**
50      * Interface for a WorkerHandler result return.
51      */
52     public interface OnImageLoadCompleteListener {
53         /**
54          * Called when the image load is complete.
55          *
56          * @param token Integer passed in {@link ContactsAsyncHelper#startObtainPhotoAsync(int,
57          * Context, Uri, OnImageLoadCompleteListener, Object)}.
58          * @param photo Drawable object obtained by the async load.
59          * @param photoIcon Bitmap object obtained by the async load.
60          * @param cookie Object passed in {@link ContactsAsyncHelper#startObtainPhotoAsync(int,
61          * Context, Uri, OnImageLoadCompleteListener, Object)}. Can be null iff. the original
62          * cookie is null.
63          */
64         public void onImageLoadComplete(int token, Drawable photo, Bitmap photoIcon,
65                 Object cookie);
66     }
67 
68     /**
69      * Interface to enable stubbing of the call to openInputStream
70      */
71     public interface ContentResolverAdapter {
72         InputStream openInputStream(Context context, Uri uri) throws FileNotFoundException;
73     }
74 
75     // constants
76     private static final int EVENT_LOAD_IMAGE = 1;
77 
78     /** Handler run on a worker thread to load photo asynchronously. */
79     private Handler mThreadHandler;
80     private final ContentResolverAdapter mContentResolverAdapter;
81 
82     public ContactsAsyncHelper(ContentResolverAdapter contentResolverAdapter) {
83         mContentResolverAdapter = contentResolverAdapter;
84     }
85 
86     public ContactsAsyncHelper(ContentResolverAdapter contentResolverAdapter, Looper looper) {
87         mContentResolverAdapter = contentResolverAdapter;
88         mThreadHandler = new WorkerHandler(looper);
89     }
90 
91     private static final class WorkerArgs {
92         public Context context;
93         public Uri displayPhotoUri;
94         public Drawable photo;
95         public Bitmap photoIcon;
96         public Object cookie;
97         public OnImageLoadCompleteListener listener;
98     }
99 
100     /**
101      * Thread worker class that handles the task of opening the stream and loading
102      * the images.
103      */
104     private class WorkerHandler extends Handler {
105         public WorkerHandler(Looper looper) {
106             super(looper);
107         }
108 
109         @Override
110         public void handleMessage(Message msg) {
111             WorkerArgs args = (WorkerArgs) msg.obj;
112 
113             switch (msg.arg1) {
114                 case EVENT_LOAD_IMAGE:
115                     InputStream inputStream = null;
116                     try {
117                         try {
118                             inputStream = mContentResolverAdapter.openInputStream(
119                                     args.context, args.displayPhotoUri);
120                         } catch (Exception e) {
121                             Log.e(this, e, "Error opening photo input stream");
122                         }
123 
124                         if (inputStream != null) {
125                             args.photo = Drawable.createFromStream(inputStream,
126                                     args.displayPhotoUri.toString());
127 
128                             // This assumes Drawable coming from contact database is usually
129                             // BitmapDrawable and thus we can have (down)scaled version of it.
130                             args.photoIcon = getPhotoIconWhenAppropriate(args.context, args.photo);
131 
132                             Log.d(this, "Loading image: " + msg.arg1 +
133                                     " token: " + msg.what + " image URI: " + args.displayPhotoUri);
134                         } else {
135                             args.photo = null;
136                             args.photoIcon = null;
137                             Log.d(this, "Problem with image: " + msg.arg1 +
138                                     " token: " + msg.what + " image URI: " + args.displayPhotoUri +
139                                     ", using default image.");
140                         }
141                     } finally {
142                         if (inputStream != null) {
143                             try {
144                                 inputStream.close();
145                             } catch (IOException e) {
146                                 Log.e(this, e, "Unable to close input stream.");
147                             }
148                         }
149                     }
150 
151                     // Listener will synchronize as needed
152                     Log.d(this, "Notifying listener: " + args.listener.toString() +
153                             " image: " + args.displayPhotoUri + " completed");
154                     args.listener.onImageLoadComplete(msg.what, args.photo, args.photoIcon,
155                                 args.cookie);
156                     break;
157                 default:
158                     break;
159             }
160         }
161 
162         /**
163          * Returns a Bitmap object suitable for {@link Notification}'s large icon. This might
164          * return null when the given Drawable isn't BitmapDrawable, or if the system fails to
165          * create a scaled Bitmap for the Drawable.
166          */
167         private Bitmap getPhotoIconWhenAppropriate(Context context, Drawable photo) {
168             if (!(photo instanceof BitmapDrawable)) {
169                 return null;
170             }
171             int iconSize = context.getResources()
172                     .getDimensionPixelSize(R.dimen.notification_icon_size);
173             Bitmap orgBitmap = ((BitmapDrawable) photo).getBitmap();
174             int orgWidth = orgBitmap.getWidth();
175             int orgHeight = orgBitmap.getHeight();
176             int longerEdge = orgWidth > orgHeight ? orgWidth : orgHeight;
177             // We want downscaled one only when the original icon is too big.
178             if (longerEdge > iconSize) {
179                 float ratio = ((float) longerEdge) / iconSize;
180                 int newWidth = (int) (orgWidth / ratio);
181                 int newHeight = (int) (orgHeight / ratio);
182                 // If the longer edge is much longer than the shorter edge, the latter may
183                 // become 0 which will cause a crash.
184                 if (newWidth <= 0 || newHeight <= 0) {
185                     Log.w(this, "Photo icon's width or height become 0.");
186                     return null;
187                 }
188 
189                 // It is sure ratio >= 1.0f in any case and thus the newly created Bitmap
190                 // should be smaller than the original.
191                 return Bitmap.createScaledBitmap(orgBitmap, newWidth, newHeight, true);
192             } else {
193                 return orgBitmap;
194             }
195         }
196     }
197 
198     /**
199      * Starts an asynchronous image load. After finishing the load,
200      * {@link OnImageLoadCompleteListener#onImageLoadComplete(int, Drawable, Bitmap, Object)}
201      * will be called.
202      *
203      * @param token Arbitrary integer which will be returned as the first argument of
204      * {@link OnImageLoadCompleteListener#onImageLoadComplete(int, Drawable, Bitmap, Object)}
205      * @param context Context object used to do the time-consuming operation.
206      * @param displayPhotoUri Uri to be used to fetch the photo
207      * @param listener Callback object which will be used when the asynchronous load is done.
208      * Can be null, which means only the asynchronous load is done while there's no way to
209      * obtain the loaded photos.
210      * @param cookie Arbitrary object the caller wants to remember, which will become the
211      * fourth argument of {@link OnImageLoadCompleteListener#onImageLoadComplete(int, Drawable,
212      * Bitmap, Object)}. Can be null, at which the callback will also has null for the argument.
213      */
214     public void startObtainPhotoAsync(int token, Context context, Uri displayPhotoUri,
215             OnImageLoadCompleteListener listener, Object cookie) {
216         ensureAsyncHandlerStarted();
217 
218         // in case the source caller info is null, the URI will be null as well.
219         // just update using the placeholder image in this case.
220         if (displayPhotoUri == null) {
221             Log.wtf(LOG_TAG, "Uri is missing");
222             return;
223         }
224 
225         // Added additional Cookie field in the callee to handle arguments
226         // sent to the callback function.
227 
228         // setup arguments
229         WorkerArgs args = new WorkerArgs();
230         args.cookie = cookie;
231         args.context = context;
232         args.displayPhotoUri = displayPhotoUri;
233         args.listener = listener;
234 
235         // setup message arguments
236         Message msg = mThreadHandler.obtainMessage(token);
237         msg.arg1 = EVENT_LOAD_IMAGE;
238         msg.obj = args;
239 
240         Log.d(LOG_TAG, "Begin loading image: " + args.displayPhotoUri +
241                 ", displaying default image for now.");
242 
243         // notify the thread to begin working
244         mThreadHandler.sendMessage(msg);
245     }
246 
247     private void ensureAsyncHandlerStarted() {
248         if (mThreadHandler == null) {
249             HandlerThread thread = new HandlerThread("ContactsAsyncWorker");
250             thread.start();
251             mThreadHandler = new WorkerHandler(thread.getLooper());
252         }
253     }
254 }
255