OSDN Git Service

Merge "Accelerate PNG image rendering." into gb-ub-photos-arches
[android-x86/packages-apps-Gallery2.git] / src / com / android / gallery3d / util / GalleryUtils.java
1 /*
2  * Copyright (C) 2010 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.gallery3d.util;
18
19 import android.annotation.TargetApi;
20 import android.content.ActivityNotFoundException;
21 import android.content.ComponentName;
22 import android.content.Context;
23 import android.content.Intent;
24 import android.content.SharedPreferences;
25 import android.content.pm.PackageManager;
26 import android.content.pm.ResolveInfo;
27 import android.content.res.Resources;
28 import android.graphics.Color;
29 import android.net.Uri;
30 import android.os.ConditionVariable;
31 import android.os.Environment;
32 import android.os.StatFs;
33 import android.preference.PreferenceManager;
34 import android.provider.MediaStore;
35 import android.util.DisplayMetrics;
36 import android.util.Log;
37 import android.view.WindowManager;
38
39 import com.android.gallery3d.R;
40 import com.android.gallery3d.app.Gallery;
41 import com.android.gallery3d.app.PackagesMonitor;
42 import com.android.gallery3d.common.ApiHelper;
43 import com.android.gallery3d.data.DataManager;
44 import com.android.gallery3d.data.MediaItem;
45 import com.android.gallery3d.ui.BitmapScreenNail;
46 import com.android.gallery3d.util.ThreadPool.CancelListener;
47 import com.android.gallery3d.util.ThreadPool.JobContext;
48
49 import java.util.Arrays;
50 import java.util.List;
51 import java.util.Locale;
52
53 public class GalleryUtils {
54     private static final String TAG = "GalleryUtils";
55     private static final String MAPS_PACKAGE_NAME = "com.google.android.apps.maps";
56     private static final String MAPS_CLASS_NAME = "com.google.android.maps.MapsActivity";
57     private static final String CAMERA_LAUNCHER_NAME = "com.android.camera.CameraLauncher";
58
59     public static final String MIME_TYPE_IMAGE = "image/*";
60     public static final String MIME_TYPE_VIDEO = "video/*";
61     public static final String MIME_TYPE_PANORAMA = "application/vnd.google.panorama360+jpg";
62     public static final String MIME_TYPE_ALL = "*/*";
63
64     private static final String DIR_TYPE_IMAGE = "vnd.android.cursor.dir/image";
65     private static final String DIR_TYPE_VIDEO = "vnd.android.cursor.dir/video";
66
67     private static final String PREFIX_PHOTO_EDITOR_UPDATE = "editor-update-";
68     private static final String PREFIX_HAS_PHOTO_EDITOR = "has-editor-";
69
70     private static final String KEY_CAMERA_UPDATE = "camera-update";
71     private static final String KEY_HAS_CAMERA = "has-camera";
72
73     private static float sPixelDensity = -1f;
74     private static boolean sCameraAvailableInitialized = false;
75     private static boolean sCameraAvailable;
76
77     public static void initialize(Context context) {
78         if (sPixelDensity < 0) {
79             DisplayMetrics metrics = new DisplayMetrics();
80             WindowManager wm = (WindowManager)
81                     context.getSystemService(Context.WINDOW_SERVICE);
82             wm.getDefaultDisplay().getMetrics(metrics);
83             sPixelDensity = metrics.density;
84         }
85         Resources r = context.getResources();
86         BitmapScreenNail.setPlaceholderColor(r.getColor(
87                 R.color.bitmap_screennail_placeholder));
88     }
89
90     public static boolean isHighResolution(Context context) {
91         DisplayMetrics metrics = new DisplayMetrics();
92         WindowManager wm = (WindowManager)
93                 context.getSystemService(Context.WINDOW_SERVICE);
94         wm.getDefaultDisplay().getMetrics(metrics);
95         return metrics.heightPixels > 2048 ||  metrics.widthPixels > 2048;
96     }
97
98     public static float[] intColorToFloatARGBArray(int from) {
99         return new float[] {
100             Color.alpha(from) / 255f,
101             Color.red(from) / 255f,
102             Color.green(from) / 255f,
103             Color.blue(from) / 255f
104         };
105     }
106
107     public static float dpToPixel(float dp) {
108         return sPixelDensity * dp;
109     }
110
111     public static int dpToPixel(int dp) {
112         return Math.round(dpToPixel((float) dp));
113     }
114
115     public static int meterToPixel(float meter) {
116         // 1 meter = 39.37 inches, 1 inch = 160 dp.
117         return Math.round(dpToPixel(meter * 39.37f * 160));
118     }
119
120     public static byte[] getBytes(String in) {
121         byte[] result = new byte[in.length() * 2];
122         int output = 0;
123         for (char ch : in.toCharArray()) {
124             result[output++] = (byte) (ch & 0xFF);
125             result[output++] = (byte) (ch >> 8);
126         }
127         return result;
128     }
129
130     // Below are used the detect using database in the render thread. It only
131     // works most of the time, but that's ok because it's for debugging only.
132
133     private static volatile Thread sCurrentThread;
134     private static volatile boolean sWarned;
135
136     public static void setRenderThread() {
137         sCurrentThread = Thread.currentThread();
138     }
139
140     public static void assertNotInRenderThread() {
141         if (!sWarned) {
142             if (Thread.currentThread() == sCurrentThread) {
143                 sWarned = true;
144                 Log.w(TAG, new Throwable("Should not do this in render thread"));
145             }
146         }
147     }
148
149     private static final double RAD_PER_DEG = Math.PI / 180.0;
150     private static final double EARTH_RADIUS_METERS = 6367000.0;
151
152     public static double fastDistanceMeters(double latRad1, double lngRad1,
153             double latRad2, double lngRad2) {
154        if ((Math.abs(latRad1 - latRad2) > RAD_PER_DEG)
155              || (Math.abs(lngRad1 - lngRad2) > RAD_PER_DEG)) {
156            return accurateDistanceMeters(latRad1, lngRad1, latRad2, lngRad2);
157        }
158        // Approximate sin(x) = x.
159        double sineLat = (latRad1 - latRad2);
160
161        // Approximate sin(x) = x.
162        double sineLng = (lngRad1 - lngRad2);
163
164        // Approximate cos(lat1) * cos(lat2) using
165        // cos((lat1 + lat2)/2) ^ 2
166        double cosTerms = Math.cos((latRad1 + latRad2) / 2.0);
167        cosTerms = cosTerms * cosTerms;
168        double trigTerm = sineLat * sineLat + cosTerms * sineLng * sineLng;
169        trigTerm = Math.sqrt(trigTerm);
170
171        // Approximate arcsin(x) = x
172        return EARTH_RADIUS_METERS * trigTerm;
173     }
174
175     public static double accurateDistanceMeters(double lat1, double lng1,
176             double lat2, double lng2) {
177         double dlat = Math.sin(0.5 * (lat2 - lat1));
178         double dlng = Math.sin(0.5 * (lng2 - lng1));
179         double x = dlat * dlat + dlng * dlng * Math.cos(lat1) * Math.cos(lat2);
180         return (2 * Math.atan2(Math.sqrt(x), Math.sqrt(Math.max(0.0,
181                 1.0 - x)))) * EARTH_RADIUS_METERS;
182     }
183
184
185     public static final double toMile(double meter) {
186         return meter / 1609;
187     }
188
189     // For debugging, it will block the caller for timeout millis.
190     public static void fakeBusy(JobContext jc, int timeout) {
191         final ConditionVariable cv = new ConditionVariable();
192         jc.setCancelListener(new CancelListener() {
193             @Override
194             public void onCancel() {
195                 cv.open();
196             }
197         });
198         cv.block(timeout);
199         jc.setCancelListener(null);
200     }
201
202     public static boolean isEditorAvailable(Context context, String mimeType) {
203         int version = PackagesMonitor.getPackagesVersion(context);
204
205         String updateKey = PREFIX_PHOTO_EDITOR_UPDATE + mimeType;
206         String hasKey = PREFIX_HAS_PHOTO_EDITOR + mimeType;
207
208         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
209         if (prefs.getInt(updateKey, 0) != version) {
210             PackageManager packageManager = context.getPackageManager();
211             List<ResolveInfo> infos = packageManager.queryIntentActivities(
212                     new Intent(Intent.ACTION_EDIT).setType(mimeType), 0);
213             prefs.edit().putInt(updateKey, version)
214                         .putBoolean(hasKey, !infos.isEmpty())
215                         .commit();
216         }
217
218         return prefs.getBoolean(hasKey, true);
219     }
220
221     public static boolean isAnyCameraAvailable(Context context) {
222         int version = PackagesMonitor.getPackagesVersion(context);
223         SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
224         if (prefs.getInt(KEY_CAMERA_UPDATE, 0) != version) {
225             PackageManager packageManager = context.getPackageManager();
226             List<ResolveInfo> infos = packageManager.queryIntentActivities(
227                     new Intent(MediaStore.INTENT_ACTION_STILL_IMAGE_CAMERA), 0);
228             prefs.edit().putInt(KEY_CAMERA_UPDATE, version)
229                         .putBoolean(KEY_HAS_CAMERA, !infos.isEmpty())
230                         .commit();
231         }
232         return prefs.getBoolean(KEY_HAS_CAMERA, true);
233     }
234
235     public static boolean isCameraAvailable(Context context) {
236         if (sCameraAvailableInitialized) return sCameraAvailable;
237         PackageManager pm = context.getPackageManager();
238         ComponentName name = new ComponentName(context, CAMERA_LAUNCHER_NAME);
239         int state = pm.getComponentEnabledSetting(name);
240         sCameraAvailableInitialized = true;
241         sCameraAvailable =
242             (state == PackageManager.COMPONENT_ENABLED_STATE_DEFAULT)
243              || (state == PackageManager.COMPONENT_ENABLED_STATE_ENABLED);
244         return sCameraAvailable;
245     }
246
247     public static void startCameraActivity(Context context) {
248         Intent intent = new Intent(MediaStore.INTENT_ACTION_STILL_IMAGE_CAMERA)
249                 .setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP
250                         | Intent.FLAG_ACTIVITY_NEW_TASK);
251         context.startActivity(intent);
252     }
253
254     public static void startGalleryActivity(Context context) {
255         Intent intent = new Intent(context, Gallery.class);
256         context.startActivity(intent);
257     }
258
259     public static boolean isValidLocation(double latitude, double longitude) {
260         // TODO: change || to && after we fix the default location issue
261         return (latitude != MediaItem.INVALID_LATLNG || longitude != MediaItem.INVALID_LATLNG);
262     }
263
264     public static String formatLatitudeLongitude(String format, double latitude,
265             double longitude) {
266         // We need to specify the locale otherwise it may go wrong in some language
267         // (e.g. Locale.FRENCH)
268         return String.format(Locale.ENGLISH, format, latitude, longitude);
269     }
270
271     public static void showOnMap(Context context, double latitude, double longitude) {
272         try {
273             // We don't use "geo:latitude,longitude" because it only centers
274             // the MapView to the specified location, but we need a marker
275             // for further operations (routing to/from).
276             // The q=(lat, lng) syntax is suggested by geo-team.
277             String uri = formatLatitudeLongitude("http://maps.google.com/maps?f=q&q=(%f,%f)",
278                     latitude, longitude);
279             ComponentName compName = new ComponentName(MAPS_PACKAGE_NAME,
280                     MAPS_CLASS_NAME);
281             Intent mapsIntent = new Intent(Intent.ACTION_VIEW,
282                     Uri.parse(uri)).setComponent(compName);
283             context.startActivity(mapsIntent);
284         } catch (ActivityNotFoundException e) {
285             // Use the "geo intent" if no GMM is installed
286             Log.e(TAG, "GMM activity not found!", e);
287             String url = formatLatitudeLongitude("geo:%f,%f", latitude, longitude);
288             Intent mapsIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url));
289             context.startActivity(mapsIntent);
290         }
291     }
292
293     public static void setViewPointMatrix(
294             float matrix[], float x, float y, float z) {
295         // The matrix is
296         // -z,  0,  x,  0
297         //  0, -z,  y,  0
298         //  0,  0,  1,  0
299         //  0,  0,  1, -z
300         Arrays.fill(matrix, 0, 16, 0);
301         matrix[0] = matrix[5] = matrix[15] = -z;
302         matrix[8] = x;
303         matrix[9] = y;
304         matrix[10] = matrix[11] = 1;
305     }
306
307     public static int getBucketId(String path) {
308         return path.toLowerCase().hashCode();
309     }
310
311     // Returns a (localized) string for the given duration (in seconds).
312     public static String formatDuration(final Context context, int duration) {
313         int h = duration / 3600;
314         int m = (duration - h * 3600) / 60;
315         int s = duration - (h * 3600 + m * 60);
316         String durationValue;
317         if (h == 0) {
318             durationValue = String.format(context.getString(R.string.details_ms), m, s);
319         } else {
320             durationValue = String.format(context.getString(R.string.details_hms), h, m, s);
321         }
322         return durationValue;
323     }
324
325     @TargetApi(ApiHelper.VERSION_CODES.HONEYCOMB)
326     public static int determineTypeBits(Context context, Intent intent) {
327         int typeBits = 0;
328         String type = intent.resolveType(context);
329
330         if (MIME_TYPE_ALL.equals(type)) {
331             typeBits = DataManager.INCLUDE_ALL;
332         } else if (MIME_TYPE_IMAGE.equals(type) ||
333                 DIR_TYPE_IMAGE.equals(type)) {
334             typeBits = DataManager.INCLUDE_IMAGE;
335         } else if (MIME_TYPE_VIDEO.equals(type) ||
336                 DIR_TYPE_VIDEO.equals(type)) {
337             typeBits = DataManager.INCLUDE_VIDEO;
338         } else {
339             typeBits = DataManager.INCLUDE_ALL;
340         }
341
342         if (ApiHelper.HAS_INTENT_EXTRA_LOCAL_ONLY) {
343             if (intent.getBooleanExtra(Intent.EXTRA_LOCAL_ONLY, false)) {
344                 typeBits |= DataManager.INCLUDE_LOCAL_ONLY;
345             }
346         }
347
348         return typeBits;
349     }
350
351     public static int getSelectionModePrompt(int typeBits) {
352         if ((typeBits & DataManager.INCLUDE_VIDEO) != 0) {
353             return (typeBits & DataManager.INCLUDE_IMAGE) == 0
354                     ? R.string.select_video
355                     : R.string.select_item;
356         }
357         return R.string.select_image;
358     }
359
360     public static boolean hasSpaceForSize(long size) {
361         String state = Environment.getExternalStorageState();
362         if (!Environment.MEDIA_MOUNTED.equals(state)) {
363             return false;
364         }
365
366         String path = Environment.getExternalStorageDirectory().getPath();
367         try {
368             StatFs stat = new StatFs(path);
369             return stat.getAvailableBlocks() * (long) stat.getBlockSize() > size;
370         } catch (Exception e) {
371             Log.i(TAG, "Fail to access external storage", e);
372         }
373         return false;
374     }
375
376     public static boolean isPanorama(MediaItem item) {
377         if (item == null) return false;
378         int w = item.getWidth();
379         int h = item.getHeight();
380         return (h > 0 && w / h >= 2);
381     }
382 }