Android2.2下camera应用程序支持GPS信息写入jpeg文件

时间:2023-02-07 09:26:14
 

一、概述

在Android2.2中,Camera的应用程序并不支持将GPS信息写入到JPEG文件中,但如果要实现这个功能,有如下两种方式:

1、修改底层camera驱动。在拍照时,一般都是使用硬件去进行JPEG编码,这样就需要修改JPEG编码器,使其可以将GPS信息写入JPEG文件的头部,即EXIF部分。这种方式使用与手机驱动开发者。

2、修改camera应用程序。Camera应用程序本身不支持该功能,但是android系统中提供了支持该功能的类——ExifInterface。本文介绍如何使用该类进行GPS信息的写入。这种方法的不足在于,每次写入GPS功能,都会把原有的JPEG文件读出,修改了Exif header部分后再写入文件。

二、实现GPS写入功能

首先来看看文件ImageManager.java,该文件位于:

/package/apps/Camera/src/com/android/camera/

该文件中,有个addImage()函数,其定义为:

public static Uri addImage(ContentResolver cr, String title, long dateTaken,

            Location location, String directory, String filename,

            Bitmap source, byte[] jpegData, int[] degree) {

        。。。。。。

        String filePath = directory + "/" + filename;

        。。。。。。

        if (location != null) {

            values.put(Images.Media.LATITUDE, location.getLatitude());

            values.put(Images.Media.LONGITUDE, location.getLongitude());        

           }

       }

    return cr.insert(STORAGE_URI, values);

}

此处,当location不等于null时,表示已经开启存储位置的功能,并且该手机的GPS功能已开启并且正常。在这里,我们就可以把GPS的信息写入JPEG文件中。其具体code如下:

public static Uri addImage(ContentResolver cr, String title, long dateTaken,

            Location location, String directory, String filename,

            Bitmap source, byte[] jpegData, int[] degree) {

        。。。。。。

        String filePath = directory + "/" + filename;

        。。。。。。

        if (location != null) {

            values.put(Images.Media.LATITUDE, location.getLatitude());

            values.put(Images.Media.LONGITUDE, location.getLongitude());

           ExifInterface exif = null;

        try {

            exif = new ExifInterface(filePath);

        } catch (IOException ex) {

            Log.e(TAG, "cannot read exif", ex);

           }

         exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE, Double.toString(location.getLatitude()));        

           exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE, Double.toString(location.getLongitude()));

           try{

               if(exif != null)

              exif.saveAttributes();

           } catch (IOException ex) {

              Log.e(TAG, "Fail to exif.saveAttributes().");

           }

       }

    return cr.insert(STORAGE_URI, values);

}

 

三、分析GPS写入功能的实现

首先看看类ExifInterface的构造函数,其位于:

/framework/base/media/java/android/media/ ExifInterface.java

其具体实现为:

    public ExifInterface(String filename) throws IOException {

        mFilename = filename;

        loadAttributes();

}

其功能是从指定的文件中获取其Exif信息。函数loadAttributes()的定义为:

    private void loadAttributes() throws IOException {

        // 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"

        mAttributes = new HashMap<String, String>();

 

        String attrStr;

        synchronized (sLock) {

            attrStr = getAttributesNative(mFilename);

        }

        ……

}

该函数从文件中读取Exif信息,并将其写入mAttributes中。函数

getAttributesNative(mFilename),调用了JNI接口,其定义位于:

/external/jhead/main.c

static JNINativeMethod methods[] = {

  {"saveAttributesNative", "(Ljava/lang/String;Ljava/lang/String;)V", (void*)saveAttributes },

  {"getAttributesNative", "(Ljava/lang/String;)Ljava/lang/String;", (void*)getAttributes },

  {"appendThumbnailNative", "(Ljava/lang/String;Ljava/lang/String;)Z", (void*)appendThumbnail },

  {"commitChangesNative", "(Ljava/lang/String;)V", (void*)commitChanges },

  {"getThumbnailNative", "(Ljava/lang/String;)[B", (void*)getThumbnail },

};

 

函数setAttribute()的实现如下:

    public void setAttribute(String tag, String value) {

        mAttributes.put(tag, value);

}

向mAttributes写入对应的项,比如经度和纬度信息。

最重要的函数saveAttributes(),它也是调用JNI接口。它负责将所有的Exif项写入到JPEG文件中。由于时间关系,就不做介绍了,具体代码请大家自己看,有问题的话,一起讨论。