1 /* 2 * Copyright (C) 2007 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 android.media; 18 19 import java.io.IOException; 20 import java.text.ParsePosition; 21 import java.text.SimpleDateFormat; 22 import java.util.Date; 23 import java.util.HashMap; 24 import java.util.Map; 25 import java.util.TimeZone; 26 27 /** 28 * This is a class for reading and writing Exif tags in a JPEG file. 29 */ 30 public class ExifInterface { 31 // The Exif tag names 32 /** Type is int. */ 33 public static final String TAG_ORIENTATION = "Orientation"; 34 /** Type is String. */ 35 public static final String TAG_DATETIME = "DateTime"; 36 /** Type is String. */ 37 public static final String TAG_MAKE = "Make"; 38 /** Type is String. */ 39 public static final String TAG_MODEL = "Model"; 40 /** Type is int. */ 41 public static final String TAG_FLASH = "Flash"; 42 /** Type is int. */ 43 public static final String TAG_IMAGE_WIDTH = "ImageWidth"; 44 /** Type is int. */ 45 public static final String TAG_IMAGE_LENGTH = "ImageLength"; 46 /** String. Format is "num1/denom1,num2/denom2,num3/denom3". */ 47 public static final String TAG_GPS_LATITUDE = "GPSLatitude"; 48 /** String. Format is "num1/denom1,num2/denom2,num3/denom3". */ 49 public static final String TAG_GPS_LONGITUDE = "GPSLongitude"; 50 /** Type is String. */ 51 public static final String TAG_GPS_LATITUDE_REF = "GPSLatitudeRef"; 52 /** Type is String. */ 53 public static final String TAG_GPS_LONGITUDE_REF = "GPSLongitudeRef"; 54 /** Type is String. */ 55 public static final String TAG_EXPOSURE_TIME = "ExposureTime"; 56 /** Type is String. */ 57 public static final String TAG_APERTURE = "FNumber"; 58 /** Type is String. */ 59 public static final String TAG_ISO = "ISOSpeedRatings"; 60 61 /** 62 * The altitude (in meters) based on the reference in TAG_GPS_ALTITUDE_REF. 63 * Type is rational. 64 */ 65 public static final String TAG_GPS_ALTITUDE = "GPSAltitude"; 66 67 /** 68 * 0 if the altitude is above sea level. 1 if the altitude is below sea 69 * level. Type is int. 70 */ 71 public static final String TAG_GPS_ALTITUDE_REF = "GPSAltitudeRef"; 72 73 /** Type is String. */ 74 public static final String TAG_GPS_TIMESTAMP = "GPSTimeStamp"; 75 /** Type is String. */ 76 public static final String TAG_GPS_DATESTAMP = "GPSDateStamp"; 77 /** Type is int. */ 78 public static final String TAG_WHITE_BALANCE = "WhiteBalance"; 79 /** Type is rational. */ 80 public static final String TAG_FOCAL_LENGTH = "FocalLength"; 81 /** Type is String. Name of GPS processing method used for location finding. */ 82 public static final String TAG_GPS_PROCESSING_METHOD = "GPSProcessingMethod"; 83 84 // Constants used for the Orientation Exif tag. 85 public static final int ORIENTATION_UNDEFINED = 0; 86 public static final int ORIENTATION_NORMAL = 1; 87 public static final int ORIENTATION_FLIP_HORIZONTAL = 2; // left right reversed mirror 88 public static final int ORIENTATION_ROTATE_180 = 3; 89 public static final int ORIENTATION_FLIP_VERTICAL = 4; // upside down mirror 90 public static final int ORIENTATION_TRANSPOSE = 5; // flipped about top-left <--> bottom-right axis 91 public static final int ORIENTATION_ROTATE_90 = 6; // rotate 90 cw to right it 92 public static final int ORIENTATION_TRANSVERSE = 7; // flipped about top-right <--> bottom-left axis 93 public static final int ORIENTATION_ROTATE_270 = 8; // rotate 270 to right it 94 95 // Constants used for white balance 96 public static final int WHITEBALANCE_AUTO = 0; 97 public static final int WHITEBALANCE_MANUAL = 1; 98 private static SimpleDateFormat sFormatter; 99 100 static { 101 System.loadLibrary("jhead_jni"); 102 sFormatter = new SimpleDateFormat("yyyy:MM:dd HH:mm:ss"); 103 sFormatter.setTimeZone(TimeZone.getTimeZone("UTC")); 104 } 105 106 private String mFilename; 107 private HashMap<String, String> mAttributes; 108 private boolean mHasThumbnail; 109 110 // Because the underlying implementation (jhead) uses static variables, 111 // there can only be one user at a time for the native functions (and 112 // they cannot keep state in the native code across function calls). We 113 // use sLock to serialize the accesses. 114 private static final Object sLock = new Object(); 115 116 /** 117 * Reads Exif tags from the specified JPEG file. 118 */ ExifInterface(String filename)119 public ExifInterface(String filename) throws IOException { 120 if (filename == null) { 121 throw new IllegalArgumentException("filename cannot be null"); 122 } 123 mFilename = filename; 124 loadAttributes(); 125 } 126 127 /** 128 * Returns the value of the specified tag or {@code null} if there 129 * is no such tag in the JPEG file. 130 * 131 * @param tag the name of the tag. 132 */ getAttribute(String tag)133 public String getAttribute(String tag) { 134 return mAttributes.get(tag); 135 } 136 137 /** 138 * Returns the integer value of the specified tag. If there is no such tag 139 * in the JPEG file or the value cannot be parsed as integer, return 140 * <var>defaultValue</var>. 141 * 142 * @param tag the name of the tag. 143 * @param defaultValue the value to return if the tag is not available. 144 */ getAttributeInt(String tag, int defaultValue)145 public int getAttributeInt(String tag, int defaultValue) { 146 String value = mAttributes.get(tag); 147 if (value == null) return defaultValue; 148 try { 149 return Integer.valueOf(value); 150 } catch (NumberFormatException ex) { 151 return defaultValue; 152 } 153 } 154 155 /** 156 * Returns the double value of the specified rational tag. If there is no 157 * such tag in the JPEG file or the value cannot be parsed as double, return 158 * <var>defaultValue</var>. 159 * 160 * @param tag the name of the tag. 161 * @param defaultValue the value to return if the tag is not available. 162 */ getAttributeDouble(String tag, double defaultValue)163 public double getAttributeDouble(String tag, double defaultValue) { 164 String value = mAttributes.get(tag); 165 if (value == null) return defaultValue; 166 try { 167 int index = value.indexOf("/"); 168 if (index == -1) return defaultValue; 169 double denom = Double.parseDouble(value.substring(index + 1)); 170 if (denom == 0) return defaultValue; 171 double num = Double.parseDouble(value.substring(0, index)); 172 return num / denom; 173 } catch (NumberFormatException ex) { 174 return defaultValue; 175 } 176 } 177 178 /** 179 * Set the value of the specified tag. 180 * 181 * @param tag the name of the tag. 182 * @param value the value of the tag. 183 */ setAttribute(String tag, String value)184 public void setAttribute(String tag, String value) { 185 mAttributes.put(tag, value); 186 } 187 188 /** 189 * Initialize mAttributes with the attributes from the file mFilename. 190 * 191 * mAttributes is a HashMap which stores the Exif attributes of the file. 192 * The key is the standard tag name and the value is the tag's value: e.g. 193 * Model -> Nikon. Numeric values are stored as strings. 194 * 195 * This function also initialize mHasThumbnail to indicate whether the 196 * file has a thumbnail inside. 197 */ loadAttributes()198 private void loadAttributes() throws IOException { 199 // format of string passed from native C code: 200 // "attrCnt attr1=valueLen value1attr2=value2Len value2..." 201 // example: 202 // "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO" 203 mAttributes = new HashMap<String, String>(); 204 205 String attrStr; 206 synchronized (sLock) { 207 attrStr = getAttributesNative(mFilename); 208 } 209 210 // get count 211 int ptr = attrStr.indexOf(' '); 212 int count = Integer.parseInt(attrStr.substring(0, ptr)); 213 // skip past the space between item count and the rest of the attributes 214 ++ptr; 215 216 for (int i = 0; i < count; i++) { 217 // extract the attribute name 218 int equalPos = attrStr.indexOf('=', ptr); 219 String attrName = attrStr.substring(ptr, equalPos); 220 ptr = equalPos + 1; // skip past = 221 222 // extract the attribute value length 223 int lenPos = attrStr.indexOf(' ', ptr); 224 int attrLen = Integer.parseInt(attrStr.substring(ptr, lenPos)); 225 ptr = lenPos + 1; // skip pas the space 226 227 // extract the attribute value 228 String attrValue = attrStr.substring(ptr, ptr + attrLen); 229 ptr += attrLen; 230 231 if (attrName.equals("hasThumbnail")) { 232 mHasThumbnail = attrValue.equalsIgnoreCase("true"); 233 } else { 234 mAttributes.put(attrName, attrValue); 235 } 236 } 237 } 238 239 /** 240 * Save the tag data into the JPEG file. This is expensive because it involves 241 * copying all the JPG data from one file to another and deleting the old file 242 * and renaming the other. It's best to use {@link #setAttribute(String,String)} 243 * to set all attributes to write and make a single call rather than multiple 244 * calls for each attribute. 245 */ saveAttributes()246 public void saveAttributes() throws IOException { 247 // format of string passed to native C code: 248 // "attrCnt attr1=valueLen value1attr2=value2Len value2..." 249 // example: 250 // "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO" 251 StringBuilder sb = new StringBuilder(); 252 int size = mAttributes.size(); 253 if (mAttributes.containsKey("hasThumbnail")) { 254 --size; 255 } 256 sb.append(size + " "); 257 for (Map.Entry<String, String> iter : mAttributes.entrySet()) { 258 String key = iter.getKey(); 259 if (key.equals("hasThumbnail")) { 260 // this is a fake attribute not saved as an exif tag 261 continue; 262 } 263 String val = iter.getValue(); 264 sb.append(key + "="); 265 sb.append(val.length() + " "); 266 sb.append(val); 267 } 268 String s = sb.toString(); 269 synchronized (sLock) { 270 saveAttributesNative(mFilename, s); 271 commitChangesNative(mFilename); 272 } 273 } 274 275 /** 276 * Returns true if the JPEG file has a thumbnail. 277 */ hasThumbnail()278 public boolean hasThumbnail() { 279 return mHasThumbnail; 280 } 281 282 /** 283 * Returns the thumbnail inside the JPEG file, or {@code null} if there is no thumbnail. 284 * The returned data is in JPEG format and can be decoded using 285 * {@link android.graphics.BitmapFactory#decodeByteArray(byte[],int,int)} 286 */ getThumbnail()287 public byte[] getThumbnail() { 288 synchronized (sLock) { 289 return getThumbnailNative(mFilename); 290 } 291 } 292 293 /** 294 * Returns the offset and length of thumbnail inside the JPEG file, or 295 * {@code null} if there is no thumbnail. 296 * 297 * @return two-element array, the offset in the first value, and length in 298 * the second, or {@code null} if no thumbnail was found. 299 * @hide 300 */ getThumbnailRange()301 public long[] getThumbnailRange() { 302 synchronized (sLock) { 303 return getThumbnailRangeNative(mFilename); 304 } 305 } 306 307 /** 308 * Stores the latitude and longitude value in a float array. The first element is 309 * the latitude, and the second element is the longitude. Returns false if the 310 * Exif tags are not available. 311 */ getLatLong(float output[])312 public boolean getLatLong(float output[]) { 313 String latValue = mAttributes.get(ExifInterface.TAG_GPS_LATITUDE); 314 String latRef = mAttributes.get(ExifInterface.TAG_GPS_LATITUDE_REF); 315 String lngValue = mAttributes.get(ExifInterface.TAG_GPS_LONGITUDE); 316 String lngRef = mAttributes.get(ExifInterface.TAG_GPS_LONGITUDE_REF); 317 318 if (latValue != null && latRef != null && lngValue != null && lngRef != null) { 319 try { 320 output[0] = convertRationalLatLonToFloat(latValue, latRef); 321 output[1] = convertRationalLatLonToFloat(lngValue, lngRef); 322 return true; 323 } catch (IllegalArgumentException e) { 324 // if values are not parseable 325 } 326 } 327 328 return false; 329 } 330 331 /** 332 * Return the altitude in meters. If the exif tag does not exist, return 333 * <var>defaultValue</var>. 334 * 335 * @param defaultValue the value to return if the tag is not available. 336 */ getAltitude(double defaultValue)337 public double getAltitude(double defaultValue) { 338 double altitude = getAttributeDouble(TAG_GPS_ALTITUDE, -1); 339 int ref = getAttributeInt(TAG_GPS_ALTITUDE_REF, -1); 340 341 if (altitude >= 0 && ref >= 0) { 342 return (double) (altitude * ((ref == 1) ? -1 : 1)); 343 } else { 344 return defaultValue; 345 } 346 } 347 348 /** 349 * Returns number of milliseconds since Jan. 1, 1970, midnight. 350 * Returns -1 if the date time information if not available. 351 * @hide 352 */ getDateTime()353 public long getDateTime() { 354 String dateTimeString = mAttributes.get(TAG_DATETIME); 355 if (dateTimeString == null) return -1; 356 357 ParsePosition pos = new ParsePosition(0); 358 try { 359 Date datetime = sFormatter.parse(dateTimeString, pos); 360 if (datetime == null) return -1; 361 return datetime.getTime(); 362 } catch (IllegalArgumentException ex) { 363 return -1; 364 } 365 } 366 367 /** 368 * Returns number of milliseconds since Jan. 1, 1970, midnight UTC. 369 * Returns -1 if the date time information if not available. 370 * @hide 371 */ getGpsDateTime()372 public long getGpsDateTime() { 373 String date = mAttributes.get(TAG_GPS_DATESTAMP); 374 String time = mAttributes.get(TAG_GPS_TIMESTAMP); 375 if (date == null || time == null) return -1; 376 377 String dateTimeString = date + ' ' + time; 378 if (dateTimeString == null) return -1; 379 380 ParsePosition pos = new ParsePosition(0); 381 try { 382 Date datetime = sFormatter.parse(dateTimeString, pos); 383 if (datetime == null) return -1; 384 return datetime.getTime(); 385 } catch (IllegalArgumentException ex) { 386 return -1; 387 } 388 } 389 convertRationalLatLonToFloat( String rationalString, String ref)390 private static float convertRationalLatLonToFloat( 391 String rationalString, String ref) { 392 try { 393 String [] parts = rationalString.split(","); 394 395 String [] pair; 396 pair = parts[0].split("/"); 397 double degrees = Double.parseDouble(pair[0].trim()) 398 / Double.parseDouble(pair[1].trim()); 399 400 pair = parts[1].split("/"); 401 double minutes = Double.parseDouble(pair[0].trim()) 402 / Double.parseDouble(pair[1].trim()); 403 404 pair = parts[2].split("/"); 405 double seconds = Double.parseDouble(pair[0].trim()) 406 / Double.parseDouble(pair[1].trim()); 407 408 double result = degrees + (minutes / 60.0) + (seconds / 3600.0); 409 if ((ref.equals("S") || ref.equals("W"))) { 410 return (float) -result; 411 } 412 return (float) result; 413 } catch (NumberFormatException e) { 414 // Some of the nubmers are not valid 415 throw new IllegalArgumentException(); 416 } catch (ArrayIndexOutOfBoundsException e) { 417 // Some of the rational does not follow the correct format 418 throw new IllegalArgumentException(); 419 } 420 } 421 appendThumbnailNative(String fileName, String thumbnailFileName)422 private native boolean appendThumbnailNative(String fileName, 423 String thumbnailFileName); 424 saveAttributesNative(String fileName, String compressedAttributes)425 private native void saveAttributesNative(String fileName, 426 String compressedAttributes); 427 getAttributesNative(String fileName)428 private native String getAttributesNative(String fileName); 429 commitChangesNative(String fileName)430 private native void commitChangesNative(String fileName); 431 getThumbnailNative(String fileName)432 private native byte[] getThumbnailNative(String fileName); 433 getThumbnailRangeNative(String fileName)434 private native long[] getThumbnailRangeNative(String fileName); 435 } 436