将EXIF数据写入使用DocumentFile类保存的图像

时间:2017-09-27 08:13:33

标签: android storage-access-framework documentfile android-storage

如果选择了设备的主内存,我需要使用正确的方案从DocumentFile或Uri获取文件而不是content://com.android.externalstorage.documents/tree/primary:的文件。 要获取图像的文件或绝对路径,我需要带文件的文件:/// storage / emulated / 0或storage / emulated / 0但是我找不到一种方法来获取正确的Uri来构建文件以将EXIF数据写入图像。

我的情景是:

  1. 用户选择保存图像的路径,该路径返回带有content://com.android.externalstorage.documents onActivityResult()的Uri。我使用treeUri.toString()将此路径保存到SharedPreferences以供稍后使用。
  2. 用户拍摄照片并使用DocumentFile.fromTreeUri(MainActivity.this, Uri.parse(uriString));
  3. 保存图像
  4. 这里我失败了,得到一个正确指向图像的文件,带有内容的Uri://不会返回现有的图像。错误的Uri应该file:///storage/emulated/并且我可以使用{{1将此Uri转换为文件}}
  5. 如何使用从SAF UI获得的Uri来获取构建文件或文件所需的Uri?

    编辑: ExifInterface Support Library是针对Android 7.1+推出的,可以使用InputStream或FileDescriptor。

    File filePath = new File(URI.create(saveDir.getUri().toString()));

    注意: ExifInterface不能用于远程InputStream,例如从HttpURLConnection返回的那些。强烈建议仅将它们与content://或file:// URIs一起使用。

    上面的代码段显然是读取数据,因为它打开一个InputStream来读取数据。我需要能够将EXIF数据写入JPEG文件。

1 个答案:

答案 0 :(得分:2)

如果Api为24或以上,则使用FileDescriptor将Exif数据写入先前已保存且已知内容的图像的答案

private void writeEXIFWithFileDescriptor(Uri uri) {

    if (Build.VERSION.SDK_INT < 24) {
        showToast("writeEXIFWithInputStream() API LOWER 24", Toast.LENGTH_SHORT);
        return;
    }

    ParcelFileDescriptor parcelFileDescriptor = null;
    try {

        parcelFileDescriptor = mContext.getContentResolver().openFileDescriptor(uri, "rw");
        FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();
        showToast("writeEXIFWithFileDescriptor(): " + fileDescriptor.toString(), Toast.LENGTH_LONG);
        ExifInterface exifInterface = new ExifInterface(fileDescriptor);
        // TODO Create  Exif Tags class to save Exif data
        exifInterface.saveAttributes();

    } catch (FileNotFoundException e) {
        showToast("File Not Found " + e.getMessage(), Toast.LENGTH_LONG);

    } catch (IOException e) {
        // Handle any errors
        e.printStackTrace();
        showToast("IOEXception " + e.getMessage(), Toast.LENGTH_LONG);
    } finally {
        if (parcelFileDescriptor != null) {
            try {
                parcelFileDescriptor.close();
            } catch (IOException ignored) {
                ignored.printStackTrace();
            }
        }
    }
}

如果Api低于24,则必须使用缓冲区文件,并在写完Exif数据后将该缓冲区文件保存到DocumentFile的实际位置。

private boolean exportImageWithEXIF(Bitmap bitmap, DocumentFile documentFile) {
        OutputStream outputStream = null;
        File bufFile = new File(Environment.getExternalStorageDirectory(), "buffer.jpg");
        long freeSpace = Environment.getExternalStorageDirectory().getFreeSpace() / 1048576;
        double bitmapSize = bitmap.getAllocationByteCount() / 1048576d;

        showToast("exportImageWithEXIF() freeSpace " + freeSpace, Toast.LENGTH_LONG);
        showToast("exportImageWithEXIF() bitmap size " + bitmapSize, Toast.LENGTH_LONG);
        try {
            outputStream = new FileOutputStream(bufFile);
            // Compress image from bitmap with JPEG extension
            if (mCameraSettings.getImageFormat().equals(Constants.IMAGE_FORMAT_JPEG)) {
                isImageSaved = bitmap.compress(CompressFormat.JPEG, mCameraSettings.getImageQuality(), outputStream);
                showToast("isImageSaved: " + isImageSaved, Toast.LENGTH_SHORT);
            }

            if (isImageSaved) {
                writeEXIFWithFile(bufFile);
            }

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } finally {
            if (outputStream != null) {
                try {
                    outputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        OutputStream os = null;
        InputStream is = null;
        try {
            int len;
            byte[] buf = new byte[4096];

            os = mContext.getContentResolver().openOutputStream(documentFile.getUri());
            is = new FileInputStream(bufFile);

            while ((len = is.read(buf)) > 0) {
                os.write(buf, 0, len);
            }

            os.close();
            is.close();

            if (bufFile != null) {
                bufFile.delete();
                bufFile = null;
            }

        } catch (IOException e) {
            e.printStackTrace();
        }

        return isImageSaved;
    }

这两种方法都可用于将Exif数据写入保存到设备内存或SD卡的图像。您还可以使用Storage Access Framework中的有效Uri将映像保存到SD卡。

我还找到了一种从内容Uri获取内存和SD卡的绝对路径的方法,但它与此问题无关,并且鼓励使用Uri而不是绝对路径并且不会导致未被注意的错误,我也无法使用绝对路径将图像保存到SD卡,只能从中读取。