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.camera; 18 19 import android.content.ContentResolver; 20 import android.content.ContentValues; 21 import android.content.Context; 22 import android.graphics.Bitmap; 23 import android.graphics.BitmapFactory; 24 import android.graphics.Point; 25 import android.location.Location; 26 import android.net.Uri; 27 import android.os.Environment; 28 import android.os.StatFs; 29 import android.provider.MediaStore.Images.Media; 30 import android.util.LruCache; 31 32 import com.android.camera.data.FilmstripItemData; 33 import com.android.camera.debug.Log; 34 import com.android.camera.exif.ExifInterface; 35 import com.android.camera.util.AndroidContext; 36 import com.android.camera.util.Size; 37 import com.google.common.base.Optional; 38 39 import java.io.File; 40 import java.io.FileNotFoundException; 41 import java.io.IOException; 42 import java.io.OutputStream; 43 import java.util.HashMap; 44 import java.util.UUID; 45 46 import javax.annotation.Nonnull; 47 48 public class Storage { 49 public final String DIRECTORY; 50 public static final String JPEG_POSTFIX = ".jpg"; 51 public static final String GIF_POSTFIX = ".gif"; 52 public static final long UNAVAILABLE = -1L; 53 public static final long PREPARING = -2L; 54 public static final long UNKNOWN_SIZE = -3L; 55 public static final long ACCESS_FAILURE = -4L; 56 public static final long LOW_STORAGE_THRESHOLD_BYTES = 50000000; 57 public static final String CAMERA_SESSION_SCHEME = "camera_session"; 58 private static final Log.Tag TAG = new Log.Tag("Storage"); 59 private static final String GOOGLE_COM = "google.com"; 60 private HashMap<Uri, Uri> sSessionsToContentUris = new HashMap<>(); 61 private HashMap<Uri, Uri> sContentUrisToSessions = new HashMap<>(); 62 private LruCache<Uri, Bitmap> sSessionsToPlaceholderBitmap = 63 // 20MB cache as an upper bound for session bitmap storage 64 new LruCache<Uri, Bitmap>(20 * 1024 * 1024) { 65 @Override 66 protected int sizeOf(Uri key, Bitmap value) { 67 return value.getByteCount(); 68 } 69 }; 70 private HashMap<Uri, Point> sSessionsToSizes = new HashMap<>(); 71 private HashMap<Uri, Integer> sSessionsToPlaceholderVersions = new HashMap<>(); 72 73 private static class Singleton { 74 private static final Storage INSTANCE = new Storage(AndroidContext.instance().get()); 75 } 76 instance()77 public static Storage instance() { 78 return Singleton.INSTANCE; 79 } 80 Storage(Context context)81 private Storage(Context context) { 82 DIRECTORY = context.getExternalFilesDir(Environment.DIRECTORY_PICTURES).getPath(); 83 } 84 85 /** 86 * Save the image with default JPEG MIME type and add it to the MediaStore. 87 * 88 * @param resolver The The content resolver to use. 89 * @param title The title of the media file. 90 * @param date The date for the media file. 91 * @param location The location of the media file. 92 * @param orientation The orientation of the media file. 93 * @param exif The EXIF info. Can be {@code null}. 94 * @param jpeg The JPEG data. 95 * @param width The width of the media file after the orientation is 96 * applied. 97 * @param height The height of the media file after the orientation is 98 * applied. 99 */ addImage(ContentResolver resolver, String title, long date, Location location, int orientation, ExifInterface exif, byte[] jpeg, int width, int height)100 public Uri addImage(ContentResolver resolver, String title, long date, 101 Location location, int orientation, ExifInterface exif, byte[] jpeg, int width, 102 int height) throws IOException { 103 104 return addImage(resolver, title, date, location, orientation, exif, jpeg, width, height, 105 FilmstripItemData.MIME_TYPE_JPEG); 106 } 107 108 /** 109 * Saves the media with a given MIME type and adds it to the MediaStore. 110 * <p> 111 * The path will be automatically generated according to the title. 112 * </p> 113 * 114 * @param resolver The The content resolver to use. 115 * @param title The title of the media file. 116 * @param data The data to save. 117 * @param date The date for the media file. 118 * @param location The location of the media file. 119 * @param orientation The orientation of the media file. 120 * @param exif The EXIF info. Can be {@code null}. 121 * @param width The width of the media file after the orientation is 122 * applied. 123 * @param height The height of the media file after the orientation is 124 * applied. 125 * @param mimeType The MIME type of the data. 126 * @return The URI of the added image, or null if the image could not be 127 * added. 128 */ addImage(ContentResolver resolver, String title, long date, Location location, int orientation, ExifInterface exif, byte[] data, int width, int height, String mimeType)129 public Uri addImage(ContentResolver resolver, String title, long date, 130 Location location, int orientation, ExifInterface exif, byte[] data, int width, 131 int height, String mimeType) throws IOException { 132 133 if (data.length > 0) { 134 Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length); 135 return addImageToMediaStore(resolver, title, date, location, orientation, data.length, 136 bitmap, width, height, mimeType, exif); 137 } 138 return null; 139 } 140 141 /** 142 * Add the entry for the media file to media store. 143 * 144 * @param resolver The The content resolver to use. 145 * @param title The title of the media file. 146 * @param date The date for the media file. 147 * @param location The location of the media file. 148 * @param orientation The orientation of the media file. 149 * @param bitmap The bitmap representation of the media to store. 150 * @param width The width of the media file after the orientation is 151 * applied. 152 * @param height The height of the media file after the orientation is 153 * applied. 154 * @param mimeType The MIME type of the data. 155 * @param exif The exif of the image. 156 * @return The content URI of the inserted media file or null, if the image 157 * could not be added. 158 */ addImageToMediaStore(ContentResolver resolver, String title, long date, Location location, int orientation, long jpegLength, Bitmap bitmap, int width, int height, String mimeType, ExifInterface exif)159 public Uri addImageToMediaStore(ContentResolver resolver, String title, long date, 160 Location location, int orientation, long jpegLength, Bitmap bitmap, int width, 161 int height, String mimeType, ExifInterface exif) { 162 // Insert into MediaStore. 163 ContentValues values = getContentValuesForData(title, date, location, mimeType, true); 164 165 Uri uri = null; 166 try { 167 uri = resolver.insert(Media.EXTERNAL_CONTENT_URI, values); 168 writeBitmap(uri, exif, bitmap, resolver); 169 } catch (Throwable th) { 170 // This can happen when the external volume is already mounted, but 171 // MediaScanner has not notify MediaProvider to add that volume. 172 // The picture is still safe and MediaScanner will find it and 173 // insert it into MediaProvider. The only problem is that the user 174 // cannot click the thumbnail to review the picture. 175 Log.e(TAG, "Failed to write MediaStore" + th); 176 if (uri != null) { 177 resolver.delete(uri, null, null); 178 } 179 } 180 return uri; 181 } 182 writeBitmap(Uri uri, ExifInterface exif, Bitmap bitmap, ContentResolver resolver)183 private void writeBitmap(Uri uri, ExifInterface exif, Bitmap bitmap, ContentResolver resolver) 184 throws FileNotFoundException, IOException { 185 OutputStream os = resolver.openOutputStream(uri); 186 if (exif != null) { 187 exif.writeExif(bitmap, os); 188 } else { 189 bitmap.compress(Bitmap.CompressFormat.JPEG, 90, os); 190 } 191 os.close(); 192 193 ContentValues publishValues = new ContentValues(); 194 publishValues.put(Media.IS_PENDING, 0); 195 resolver.update(uri, publishValues, null, null); 196 Log.i(TAG, "Image with uri: " + uri + " was published to the MediaStore"); 197 } 198 199 // Get a ContentValues object for the given photo data getContentValuesForData(String title, long date, Location location, String mimeType, boolean isPending)200 public ContentValues getContentValuesForData(String title, long date, Location location, 201 String mimeType, boolean isPending) { 202 203 ContentValues values = new ContentValues(11); 204 values.put(Media.TITLE, title); 205 values.put(Media.DISPLAY_NAME, title + JPEG_POSTFIX); 206 values.put(Media.DATE_TAKEN, date); 207 values.put(Media.MIME_TYPE, mimeType); 208 209 if (isPending) { 210 values.put(Media.IS_PENDING, 1); 211 } else { 212 values.put(Media.IS_PENDING, 0); 213 } 214 215 if (location != null) { 216 values.put(Media.LATITUDE, location.getLatitude()); 217 values.put(Media.LONGITUDE, location.getLongitude()); 218 } 219 return values; 220 } 221 222 /** 223 * Add a placeholder for a new image that does not exist yet. 224 * 225 * @param placeholder the placeholder image 226 * @return A new URI used to reference this placeholder 227 */ addPlaceholder(Bitmap placeholder)228 public Uri addPlaceholder(Bitmap placeholder) { 229 Uri uri = generateUniquePlaceholderUri(); 230 replacePlaceholder(uri, placeholder); 231 return uri; 232 } 233 234 /** 235 * Remove a placeholder from in memory storage. 236 */ removePlaceholder(Uri uri)237 public void removePlaceholder(Uri uri) { 238 sSessionsToSizes.remove(uri); 239 sSessionsToPlaceholderBitmap.remove(uri); 240 sSessionsToPlaceholderVersions.remove(uri); 241 } 242 243 /** 244 * Add or replace placeholder for a new image that does not exist yet. 245 * 246 * @param uri the uri of the placeholder to replace, or null if this is a 247 * new one 248 * @param placeholder the placeholder image 249 * @return A URI used to reference this placeholder 250 */ replacePlaceholder(Uri uri, Bitmap placeholder)251 public void replacePlaceholder(Uri uri, Bitmap placeholder) { 252 Log.v(TAG, "session bitmap cache size: " + sSessionsToPlaceholderBitmap.size()); 253 Point size = new Point(placeholder.getWidth(), placeholder.getHeight()); 254 sSessionsToSizes.put(uri, size); 255 sSessionsToPlaceholderBitmap.put(uri, placeholder); 256 Integer currentVersion = sSessionsToPlaceholderVersions.get(uri); 257 sSessionsToPlaceholderVersions.put(uri, currentVersion == null ? 0 : currentVersion + 1); 258 } 259 260 /** 261 * Creates an empty placeholder. 262 * 263 * @param size the size of the placeholder in pixels. 264 * @return A new URI used to reference this placeholder 265 */ 266 @Nonnull addEmptyPlaceholder(@onnull Size size)267 public Uri addEmptyPlaceholder(@Nonnull Size size) { 268 Uri uri = generateUniquePlaceholderUri(); 269 sSessionsToSizes.put(uri, new Point(size.getWidth(), size.getHeight())); 270 sSessionsToPlaceholderBitmap.remove(uri); 271 Integer currentVersion = sSessionsToPlaceholderVersions.get(uri); 272 sSessionsToPlaceholderVersions.put(uri, currentVersion == null ? 0 : currentVersion + 1); 273 return uri; 274 } 275 276 /** 277 * Take jpeg bytes and add them to the media store, either replacing an existing item 278 * or a placeholder uri to replace 279 * @param imageUri The content uri or session uri of the image being updated 280 * @param resolver The content resolver to use 281 * @param title of the image 282 * @param date of the image 283 * @param location of the image 284 * @param orientation of the image 285 * @param exif of the image 286 * @param jpeg bytes of the image 287 * @param width of the image 288 * @param height of the image 289 * @param mimeType of the image 290 * @return The content uri of the newly inserted or replaced item. 291 */ updateImage(Uri imageUri, ContentResolver resolver, String title, long date, Location location, int orientation, ExifInterface exif, byte[] jpeg, int width, int height, String mimeType)292 public Uri updateImage(Uri imageUri, ContentResolver resolver, String title, long date, 293 Location location, int orientation, ExifInterface exif, 294 byte[] jpeg, int width, int height, String mimeType) throws IOException { 295 Bitmap bitmap = BitmapFactory.decodeByteArray(jpeg, 0, jpeg.length); 296 return updateImage(imageUri, resolver, title, date, location, orientation, jpeg.length, 297 bitmap, width, height, mimeType, exif); 298 } 299 generateUniquePlaceholderUri()300 private Uri generateUniquePlaceholderUri() { 301 Uri.Builder builder = new Uri.Builder(); 302 String uuid = UUID.randomUUID().toString(); 303 builder.scheme(CAMERA_SESSION_SCHEME).authority(GOOGLE_COM).appendPath(uuid); 304 return builder.build(); 305 } 306 307 /** 308 * Renames a file. 309 * 310 * <p/> 311 * Can only be used for regular files, not directories. 312 * 313 * @param inputPath the original path of the file 314 * @param newFilePath the new path of the file 315 * @return false if rename was not successful 316 */ renameFile(File inputPath, File newFilePath)317 public boolean renameFile(File inputPath, File newFilePath) { 318 if (newFilePath.exists()) { 319 Log.e(TAG, "File path already exists: " + newFilePath.getAbsolutePath()); 320 return false; 321 } 322 if (inputPath.isDirectory()) { 323 Log.e(TAG, "Input path is directory: " + inputPath.getAbsolutePath()); 324 return false; 325 } 326 if (!createDirectoryIfNeeded(newFilePath.getAbsolutePath())) { 327 Log.e(TAG, "Failed to create parent directory for file: " + 328 newFilePath.getAbsolutePath()); 329 return false; 330 } 331 return inputPath.renameTo(newFilePath); 332 } 333 334 /** 335 * Given a file path, makes sure the directory it's in exists, and if not 336 * that it is created. 337 * 338 * @param filePath the absolute path of a file, e.g. '/foo/bar/file.jpg'. 339 * @return Whether the directory exists. If 'false' is returned, this file 340 * cannot be written to since the parent directory could not be 341 * created. 342 */ createDirectoryIfNeeded(String filePath)343 private boolean createDirectoryIfNeeded(String filePath) { 344 File parentFile = new File(filePath).getParentFile(); 345 346 // If the parent exists, return 'true' if it is a directory. If it's a 347 // file, return 'false'. 348 if (parentFile.exists()) { 349 return parentFile.isDirectory(); 350 } 351 352 // If the parent does not exists, attempt to create it and return 353 // whether creating it succeeded. 354 return parentFile.mkdirs(); 355 } 356 357 /** Updates the image values in MediaStore. */ updateImage(Uri imageUri, ContentResolver resolver, String title, long date, Location location, int orientation, int jpegLength, Bitmap bitmap, int width, int height, String mimeType, ExifInterface exif)358 private Uri updateImage(Uri imageUri, ContentResolver resolver, String title, 359 long date, Location location, int orientation, int jpegLength, 360 Bitmap bitmap, int width, int height, String mimeType, ExifInterface exif) { 361 362 Uri resultUri = imageUri; 363 if (isSessionUri(imageUri)) { 364 // If this is a session uri, then we need to add the image 365 resultUri = addImageToMediaStore(resolver, title, date, location, orientation, 366 jpegLength, bitmap, width, height, mimeType, exif); 367 sSessionsToContentUris.put(imageUri, resultUri); 368 sContentUrisToSessions.put(resultUri, imageUri); 369 } else { 370 // Update the MediaStore 371 ContentValues values = getContentValuesForData(title, date, location, mimeType, false); 372 resolver.update(imageUri, values, null, null); 373 Log.i(TAG, "Image with uri: " + imageUri + " was updated in the MediaStore"); 374 } 375 return resultUri; 376 } 377 generateFilepath(String title, String mimeType)378 private String generateFilepath(String title, String mimeType) { 379 return generateFilepath(DIRECTORY, title, mimeType); 380 } 381 generateFilepath(String directory, String title, String mimeType)382 public String generateFilepath(String directory, String title, String mimeType) { 383 String extension = null; 384 if (FilmstripItemData.MIME_TYPE_JPEG.equals(mimeType)) { 385 extension = JPEG_POSTFIX; 386 } else if (FilmstripItemData.MIME_TYPE_GIF.equals(mimeType)) { 387 extension = GIF_POSTFIX; 388 } else { 389 throw new IllegalArgumentException("Invalid mimeType: " + mimeType); 390 } 391 return (new File(directory, title + extension)).getAbsolutePath(); 392 } 393 394 /** 395 * Returns the jpeg bytes for a placeholder session 396 * 397 * @param uri the session uri to look up 398 * @return The bitmap or null 399 */ getPlaceholderForSession(Uri uri)400 public Optional<Bitmap> getPlaceholderForSession(Uri uri) { 401 return Optional.fromNullable(sSessionsToPlaceholderBitmap.get(uri)); 402 } 403 404 /** 405 * @return Whether a placeholder size for the session with the given URI 406 * exists. 407 */ containsPlaceholderSize(Uri uri)408 public boolean containsPlaceholderSize(Uri uri) { 409 return sSessionsToSizes.containsKey(uri); 410 } 411 412 /** 413 * Returns the dimensions of the placeholder image 414 * 415 * @param uri the session uri to look up 416 * @return The size 417 */ getSizeForSession(Uri uri)418 public Point getSizeForSession(Uri uri) { 419 return sSessionsToSizes.get(uri); 420 } 421 422 /** 423 * Takes a session URI and returns the finished image's content URI 424 * 425 * @param uri the uri of the session that was replaced 426 * @return The uri of the new media item, if it exists, or null. 427 */ getContentUriForSessionUri(Uri uri)428 public Uri getContentUriForSessionUri(Uri uri) { 429 return sSessionsToContentUris.get(uri); 430 } 431 432 /** 433 * Takes a content URI and returns the original Session Uri if any 434 * 435 * @param contentUri the uri of the media store content 436 * @return The session uri of the original session, if it exists, or null. 437 */ getSessionUriFromContentUri(Uri contentUri)438 public Uri getSessionUriFromContentUri(Uri contentUri) { 439 return sContentUrisToSessions.get(contentUri); 440 } 441 442 /** 443 * Determines if a URI points to a camera session 444 * 445 * @param uri the uri to check 446 * @return true if it is a session uri. 447 */ isSessionUri(Uri uri)448 public boolean isSessionUri(Uri uri) { 449 return uri.getScheme().equals(CAMERA_SESSION_SCHEME); 450 } 451 getAvailableSpace()452 public long getAvailableSpace() { 453 String state = Environment.getExternalStorageState(); 454 Log.d(TAG, "External storage state=" + state); 455 if (Environment.MEDIA_CHECKING.equals(state)) { 456 return PREPARING; 457 } 458 if (!Environment.MEDIA_MOUNTED.equals(state)) { 459 return UNAVAILABLE; 460 } 461 462 File dir = new File(DIRECTORY); 463 dir.mkdirs(); 464 if (!dir.isDirectory() || !dir.canWrite()) { 465 Log.d(TAG, DIRECTORY + " mounted, but isn't directory or cannot write"); 466 return UNAVAILABLE; 467 } 468 469 try { 470 StatFs stat = new StatFs(DIRECTORY); 471 return stat.getAvailableBlocks() * (long) stat.getBlockSize(); 472 } catch (Exception e) { 473 Log.i(TAG, "Fail to access external storage", e); 474 } 475 return UNKNOWN_SIZE; 476 } 477 478 /** 479 * OSX requires plugged-in USB storage to have path /DCIM/NNNAAAAA to be 480 * imported. This is a temporary fix for bug#1655552. 481 */ ensureOSXCompatible()482 public void ensureOSXCompatible() { 483 File nnnAAAAA = new File(DIRECTORY, "100ANDRO"); 484 if (!(nnnAAAAA.exists() || nnnAAAAA.mkdirs())) { 485 Log.e(TAG, "Failed to create " + nnnAAAAA.getPath()); 486 } 487 } 488 489 } 490