FileDocCategorySizeDatePackage
ExifInterface.javaAPI DocAndroid 1.5 API10754Wed May 06 22:42:42 BST 2009com.android.camera

ExifInterface.java

/*
 * Copyright (C) 2007 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.android.camera;

import java.util.HashMap;
import java.util.Iterator;

import android.util.Config;
import android.util.Log;

// Wrapper for native Exif library

public class ExifInterface {

    private String mFilename;

    // Constants used for the Orientation Exif tag.
    static final int ORIENTATION_UNDEFINED = 0;
    static final int ORIENTATION_NORMAL = 1;
    static final int ORIENTATION_FLIP_HORIZONTAL = 2;   // left right reversed mirror
    static final int ORIENTATION_ROTATE_180 = 3;
    static final int ORIENTATION_FLIP_VERTICAL = 4;     // upside down mirror
    static final int ORIENTATION_TRANSPOSE = 5;         // flipped about top-left <--> bottom-right axis
    static final int ORIENTATION_ROTATE_90 = 6;         // rotate 90 cw to right it
    static final int ORIENTATION_TRANSVERSE = 7;        // flipped about top-right <--> bottom-left axis
    static final int ORIENTATION_ROTATE_270 = 8;        // rotate 270 to right it

    // The Exif tag names
    static final String TAG_ORIENTATION = "Orientation";
    static final String TAG_DATE_TIME_ORIGINAL = "DateTimeOriginal";
    static final String TAG_MAKE = "Make";
    static final String TAG_MODEL = "Model";
    static final String TAG_FLASH = "Flash";
    static final String TAG_IMAGE_WIDTH = "ImageWidth";
    static final String TAG_IMAGE_LENGTH = "ImageLength";

    static final String TAG_GPS_LATITUDE = "GPSLatitude";
    static final String TAG_GPS_LONGITUDE = "GPSLongitude";
    
    static final String TAG_GPS_LATITUDE_REF = "GPSLatitudeRef";
    static final String TAG_GPS_LONGITUDE_REF = "GPSLongitudeRef";

    private boolean mSavedAttributes = false;
    private boolean mHasThumbnail = false;
    private HashMap<String, String> mCachedAttributes = null;

    static {
        System.loadLibrary("exif");
    }

    public ExifInterface(String fileName) {
        mFilename = fileName;
    }

    /**
     * Given a HashMap of Exif tags and associated values, an Exif section in the JPG file
     * is created and loaded with the tag data. saveAttributes() is expensive because it involves
     * copying all the JPG data from one file to another and deleting the old file and renaming the other.
     * It's best to collect all the attributes to write and make a single call rather than multiple
     *  calls for each attribute. You must call "commitChanges()" at some point to commit the changes.
     */
    public void saveAttributes(HashMap<String, String> attributes) {
        // format of string passed to native C code:
        // "attrCnt attr1=valueLen value1attr2=value2Len value2..."
        // example: "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO"
        StringBuilder sb = new StringBuilder();
        int size = attributes.size();
        if (attributes.containsKey("hasThumbnail")) {
            --size;
        }
        sb.append(size + " ");
        Iterator keyIterator = attributes.keySet().iterator();
        while (keyIterator.hasNext()) {
            String key = (String)keyIterator.next();
            if (key.equals("hasThumbnail")) {
                continue;       // this is a fake attribute not saved as an exif tag
            }
            String val = (String)attributes.get(key);
            sb.append(key + "=");
            sb.append(val.length() + " ");
            sb.append(val);
        }
        String s = sb.toString();
        if (android.util.Config.LOGV)
            android.util.Log.v("camera", "saving exif data: " + s);
        saveAttributesNative(mFilename, s);
        mSavedAttributes = true;
    }

    /**
     * Returns a HashMap loaded with the Exif attributes of the file. The key is the standard
     * tag name and the value is the tag's value: e.g. Model -> Nikon. Numeric values are
     * returned as strings.
     */
    public HashMap<String, String> getAttributes() {
        if (mCachedAttributes != null) {
            return mCachedAttributes;
        }
        // format of string passed from native C code:
        // "attrCnt attr1=valueLen value1attr2=value2Len value2..."
        // example: "4 attrPtr ImageLength=4 1024Model=6 FooImageWidth=4 1280Make=3 FOO"
        mCachedAttributes = new HashMap<String, String>();

        String attrStr = getAttributesNative(mFilename);

        // get count
        int ptr = attrStr.indexOf(' ');
        int count = Integer.parseInt(attrStr.substring(0, ptr));
        ++ptr;  // skip past the space between item count and the rest of the attributes

        for (int i = 0; i < count; i++) {
            // extract the attribute name
            int equalPos = attrStr.indexOf('=', ptr);
            String attrName = attrStr.substring(ptr, equalPos);
            ptr = equalPos + 1;     // skip past =

            // extract the attribute value length
            int lenPos = attrStr.indexOf(' ', ptr);
            int attrLen = Integer.parseInt(attrStr.substring(ptr, lenPos));
            ptr = lenPos + 1;       // skip pas the space

            // extract the attribute value
            String attrValue = attrStr.substring(ptr, ptr + attrLen);
            ptr += attrLen;

            if (attrName.equals("hasThumbnail")) {
                mHasThumbnail = attrValue.equalsIgnoreCase("true");
            } else {
                mCachedAttributes.put(attrName, attrValue);
            }
        }
        return mCachedAttributes;
    }

    /**
     * Given a numerical orientation, return a human-readable string describing the orientation.
     */
    static public String orientationToString(int orientation) {
        // TODO: this function needs to be localized and use string resource ids rather than strings
        String orientationString;
        switch (orientation) {
            case ORIENTATION_NORMAL:            orientationString = "Normal";   break;
            case ORIENTATION_FLIP_HORIZONTAL:   orientationString = "Flipped horizontal";   break;
            case ORIENTATION_ROTATE_180:        orientationString = "Rotated 180 degrees";   break;
            case ORIENTATION_FLIP_VERTICAL:     orientationString = "Upside down mirror";   break;
            case ORIENTATION_TRANSPOSE:         orientationString = "Transposed";   break;
            case ORIENTATION_ROTATE_90:         orientationString = "Rotated 90 degrees";   break;
            case ORIENTATION_TRANSVERSE:        orientationString = "Transversed";   break;
            case ORIENTATION_ROTATE_270:        orientationString = "Rotated 270 degrees";   break;
            default:                            orientationString = "Undefined";   break;
        }
        return orientationString;
    }

    /**
     * Copies the thumbnail data out of the filename and puts it in the Exif data associated
     * with the file used to create this object. You must call "commitChanges()" at some point
     * to commit the changes.
     */
    public boolean appendThumbnail(String thumbnailFileName) {
        if (!mSavedAttributes) {
            throw new RuntimeException("Must call saveAttributes before calling appendThumbnail");
        }
        mHasThumbnail = appendThumbnailNative(mFilename, thumbnailFileName);
        return mHasThumbnail;
    }

    /**
     * Saves the changes (added Exif tags, added thumbnail) to the JPG file. You have to call
     * saveAttributes() before committing the changes.
     */
    public void commitChanges() {
        if (!mSavedAttributes) {
            throw new RuntimeException("Must call saveAttributes before calling commitChanges");
        }
        commitChangesNative(mFilename);
    }

    public boolean hasThumbnail() {
        if (!mSavedAttributes) {
            getAttributes();
        }
        return mHasThumbnail;
    }

    public byte[] getThumbnail() {
        return getThumbnailNative(mFilename);
    }

    static public String convertRationalLatLonToDecimalString(String rationalString, String ref, boolean usePositiveNegative) {
        try {
            String [] parts = rationalString.split(",");

            String [] pair;
            pair = parts[0].split("/");
            int degrees = (int) (Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim()));

            pair = parts[1].split("/");
            int minutes = (int) ((Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim())));

            pair = parts[2].split("/");
            float seconds = Float.parseFloat(pair[0].trim()) / Float.parseFloat(pair[1].trim());

            float result = degrees + (minutes/60F) + (seconds/(60F*60F));
            
            String preliminaryResult = String.valueOf(result);
            if (usePositiveNegative) {
                String neg = (ref.equals("S") || ref.equals("E")) ? "-" : "";
                return neg + preliminaryResult;
            } else {
                return preliminaryResult + String.valueOf((char)186) + " " + ref; 
            }
        } catch (Exception ex) {
            // if for whatever reason we can't parse the lat long then return null
            return null;
        }
    }
        
    static public String makeLatLongString(double d) {
        d = Math.abs(d);
        
        int degrees = (int) d;
        
        double remainder = d - (double)degrees;
        int minutes = (int) (remainder * 60D);
        int seconds = (int) (((remainder * 60D) - minutes) * 60D * 1000D);  // really seconds * 1000
        
        String retVal = degrees + "/1," + minutes + "/1," + (int)seconds + "/1000";
        return retVal;
    }
    
    static public String makeLatStringRef(double lat) {
        return lat >= 0D ? "N" : "S";
    }
    
    static public String makeLonStringRef(double lon) {
        return lon >= 0D ? "W" : "E";
    }

    private native boolean appendThumbnailNative(String fileName, String thumbnailFileName);

    private native void saveAttributesNative(String fileName, String compressedAttributes);

    private native String getAttributesNative(String fileName);

    private native void commitChangesNative(String fileName);

    private native byte[] getThumbnailNative(String fileName);
}