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.TiledScreenNail; 46 import com.android.gallery3d.util.ThreadPool.CancelListener; 47 import com.android.gallery3d.util.ThreadPool.JobContext; 48 49 import java.io.File; 50 import java.util.Arrays; 51 import java.util.List; 52 import java.util.Locale; 53 54 public class GalleryUtils { 55 private static final String TAG = "GalleryUtils"; 56 private static final String MAPS_PACKAGE_NAME = "com.google.android.apps.maps"; 57 private static final String MAPS_CLASS_NAME = "com.google.android.maps.MapsActivity"; 58 private static final String CAMERA_LAUNCHER_NAME = "com.android.camera.CameraLauncher"; 59 60 public static final String MIME_TYPE_IMAGE = "image/*"; 61 public static final String MIME_TYPE_VIDEO = "video/*"; 62 public static final String MIME_TYPE_PANORAMA360 = "application/vnd.google.panorama360+jpg"; 63 public static final String MIME_TYPE_ALL = "*/*"; 64 65 private static final String DIR_TYPE_IMAGE = "vnd.android.cursor.dir/image"; 66 private static final String DIR_TYPE_VIDEO = "vnd.android.cursor.dir/video"; 67 68 private static final String PREFIX_PHOTO_EDITOR_UPDATE = "editor-update-"; 69 private static final String PREFIX_HAS_PHOTO_EDITOR = "has-editor-"; 70 71 private static final String KEY_CAMERA_UPDATE = "camera-update"; 72 private static final String KEY_HAS_CAMERA = "has-camera"; 73 74 private static float sPixelDensity = -1f; 75 private static boolean sCameraAvailableInitialized = false; 76 private static boolean sCameraAvailable; 77 initialize(Context context)78 public static void initialize(Context context) { 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 Resources r = context.getResources(); 85 TiledScreenNail.setPlaceholderColor(r.getColor( 86 R.color.bitmap_screennail_placeholder)); 87 initializeThumbnailSizes(metrics, r); 88 } 89 initializeThumbnailSizes(DisplayMetrics metrics, Resources r)90 private static void initializeThumbnailSizes(DisplayMetrics metrics, Resources r) { 91 int maxPixels = Math.max(metrics.heightPixels, metrics.widthPixels); 92 93 // For screen-nails, we never need to completely fill the screen 94 MediaItem.setThumbnailSizes(maxPixels / 2, maxPixels / 5); 95 TiledScreenNail.setMaxSide(maxPixels / 2); 96 } 97 intColorToFloatARGBArray(int from)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 dpToPixel(float dp)107 public static float dpToPixel(float dp) { 108 return sPixelDensity * dp; 109 } 110 dpToPixel(int dp)111 public static int dpToPixel(int dp) { 112 return Math.round(dpToPixel((float) dp)); 113 } 114 meterToPixel(float meter)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 getBytes(String in)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 setRenderThread()136 public static void setRenderThread() { 137 sCurrentThread = Thread.currentThread(); 138 } 139 assertNotInRenderThread()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 fastDistanceMeters(double latRad1, double lngRad1, double latRad2, double lngRad2)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 accurateDistanceMeters(double lat1, double lng1, double lat2, double lng2)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 toMile(double meter)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. fakeBusy(JobContext jc, int timeout)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 isEditorAvailable(Context context, String mimeType)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 isAnyCameraAvailable(Context context)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 isCameraAvailable(Context context)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 startCameraActivity(Context context)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 startGalleryActivity(Context context)254 public static void startGalleryActivity(Context context) { 255 Intent intent = new Intent(context, Gallery.class) 256 .setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP 257 | Intent.FLAG_ACTIVITY_NEW_TASK); 258 context.startActivity(intent); 259 } 260 isValidLocation(double latitude, double longitude)261 public static boolean isValidLocation(double latitude, double longitude) { 262 // TODO: change || to && after we fix the default location issue 263 return (latitude != MediaItem.INVALID_LATLNG || longitude != MediaItem.INVALID_LATLNG); 264 } 265 formatLatitudeLongitude(String format, double latitude, double longitude)266 public static String formatLatitudeLongitude(String format, double latitude, 267 double longitude) { 268 // We need to specify the locale otherwise it may go wrong in some language 269 // (e.g. Locale.FRENCH) 270 return String.format(Locale.ENGLISH, format, latitude, longitude); 271 } 272 showOnMap(Context context, double latitude, double longitude)273 public static void showOnMap(Context context, double latitude, double longitude) { 274 try { 275 // We don't use "geo:latitude,longitude" because it only centers 276 // the MapView to the specified location, but we need a marker 277 // for further operations (routing to/from). 278 // The q=(lat, lng) syntax is suggested by geo-team. 279 String uri = formatLatitudeLongitude("http://maps.google.com/maps?f=q&q=(%f,%f)", 280 latitude, longitude); 281 ComponentName compName = new ComponentName(MAPS_PACKAGE_NAME, 282 MAPS_CLASS_NAME); 283 Intent mapsIntent = new Intent(Intent.ACTION_VIEW, 284 Uri.parse(uri)).setComponent(compName); 285 context.startActivity(mapsIntent); 286 } catch (ActivityNotFoundException e) { 287 // Use the "geo intent" if no GMM is installed 288 Log.e(TAG, "GMM activity not found!", e); 289 String url = formatLatitudeLongitude("geo:%f,%f", latitude, longitude); 290 Intent mapsIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(url)); 291 context.startActivity(mapsIntent); 292 } 293 } 294 setViewPointMatrix( float matrix[], float x, float y, float z)295 public static void setViewPointMatrix( 296 float matrix[], float x, float y, float z) { 297 // The matrix is 298 // -z, 0, x, 0 299 // 0, -z, y, 0 300 // 0, 0, 1, 0 301 // 0, 0, 1, -z 302 Arrays.fill(matrix, 0, 16, 0); 303 matrix[0] = matrix[5] = matrix[15] = -z; 304 matrix[8] = x; 305 matrix[9] = y; 306 matrix[10] = matrix[11] = 1; 307 } 308 getBucketId(String path)309 public static int getBucketId(String path) { 310 return path.toLowerCase().hashCode(); 311 } 312 313 // Return the local path that matches the given bucketId. If no match is 314 // found, return null searchDirForPath(File dir, int bucketId)315 public static String searchDirForPath(File dir, int bucketId) { 316 File[] files = dir.listFiles(); 317 if (files != null) { 318 for (File file : files) { 319 if (file.isDirectory()) { 320 String path = file.getAbsolutePath(); 321 if (GalleryUtils.getBucketId(path) == bucketId) { 322 return path; 323 } else { 324 path = searchDirForPath(file, bucketId); 325 if (path != null) return path; 326 } 327 } 328 } 329 } 330 return null; 331 } 332 333 // Returns a (localized) string for the given duration (in seconds). formatDuration(final Context context, int duration)334 public static String formatDuration(final Context context, int duration) { 335 int h = duration / 3600; 336 int m = (duration - h * 3600) / 60; 337 int s = duration - (h * 3600 + m * 60); 338 String durationValue; 339 if (h == 0) { 340 durationValue = String.format(context.getString(R.string.details_ms), m, s); 341 } else { 342 durationValue = String.format(context.getString(R.string.details_hms), h, m, s); 343 } 344 return durationValue; 345 } 346 347 @TargetApi(ApiHelper.VERSION_CODES.HONEYCOMB) determineTypeBits(Context context, Intent intent)348 public static int determineTypeBits(Context context, Intent intent) { 349 int typeBits = 0; 350 String type = intent.resolveType(context); 351 352 if (MIME_TYPE_ALL.equals(type)) { 353 typeBits = DataManager.INCLUDE_ALL; 354 } else if (MIME_TYPE_IMAGE.equals(type) || 355 DIR_TYPE_IMAGE.equals(type)) { 356 typeBits = DataManager.INCLUDE_IMAGE; 357 } else if (MIME_TYPE_VIDEO.equals(type) || 358 DIR_TYPE_VIDEO.equals(type)) { 359 typeBits = DataManager.INCLUDE_VIDEO; 360 } else { 361 typeBits = DataManager.INCLUDE_ALL; 362 } 363 364 if (ApiHelper.HAS_INTENT_EXTRA_LOCAL_ONLY) { 365 if (intent.getBooleanExtra(Intent.EXTRA_LOCAL_ONLY, false)) { 366 typeBits |= DataManager.INCLUDE_LOCAL_ONLY; 367 } 368 } 369 370 return typeBits; 371 } 372 getSelectionModePrompt(int typeBits)373 public static int getSelectionModePrompt(int typeBits) { 374 if ((typeBits & DataManager.INCLUDE_VIDEO) != 0) { 375 return (typeBits & DataManager.INCLUDE_IMAGE) == 0 376 ? R.string.select_video 377 : R.string.select_item; 378 } 379 return R.string.select_image; 380 } 381 hasSpaceForSize(long size)382 public static boolean hasSpaceForSize(long size) { 383 String state = Environment.getExternalStorageState(); 384 if (!Environment.MEDIA_MOUNTED.equals(state)) { 385 return false; 386 } 387 388 String path = Environment.getExternalStorageDirectory().getPath(); 389 try { 390 StatFs stat = new StatFs(path); 391 return stat.getAvailableBlocks() * (long) stat.getBlockSize() > size; 392 } catch (Exception e) { 393 Log.i(TAG, "Fail to access external storage", e); 394 } 395 return false; 396 } 397 isPanorama(MediaItem item)398 public static boolean isPanorama(MediaItem item) { 399 if (item == null) return false; 400 int w = item.getWidth(); 401 int h = item.getHeight(); 402 return (h > 0 && w / h >= 2); 403 } 404 } 405