Android使用原生相机保存图像

时间:2011-01-18 09:56:20

标签: android android-camera

我想启动原生Android相机并将图像保存在指定位置。问题是在我点击照片后,预览会显示保存/放弃选项。单击保存后,相机再次启动,我捕获的图像不会保存在指定位置。而是将其保存在默认位置。实际上我需要我点击的图像的位置。这是我用来启动相机的代码。

MediaScannerConnection_MSC = null;
String fileName = String.valueOf(System.currentTimeMillis())+".jpg";
f = new File(Environment.getExternalStorageDirectory(), fileName);
_imageUri = Uri.fromFile(f);
// create new Intent
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1);
intent.putExtra(MediaStore.EXTRA_OUTPUT, _imageUri);
startActivityForResult(intent, 1);

这是从相机返回后的代码

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (requestCode == 1) {
            if (resultCode == RESULT_OK) {
                // use imageUri here to access the image

                final String imagePath = f.getAbsolutePath();
                _MSC = new MediaScannerConnection(this, new MediaScannerConnectionClient() {

                    public void onMediaScannerConnected() {
                        _MSC.scanFile(imagePath, null);
                    }

                    public void onScanCompleted(String path, Uri uri) {
                        _MSC.disconnect();
                        _MSC = null;
                    }

                });
                _MSC.connect();

            }
        }
    }

我在这里犯了什么错误

1 个答案:

答案 0 :(得分:2)

我想我有你的答案,我已经搜索过这个问题的高低,这有点棘手。

首先,我认为在某些手机上,

intent.putExtra(MediaStore.EXTRA_OUTPUT, _imageUri);
忽略

: - (

所以我最好的答案是,不要设置“EXTRA_OUTPUT”标志,而是让它正常运行并返回你的新图片URI。然后,从那里,使用:

Uri u = intent.getData();

要获取新图片的信息,然后从那里,我使用FileOutputStream将图像文件写入我想要的新位置。然后,最后,我删除原始文件。 : - )

我知道这有点好,但它确实有效,而且我认为它非常可靠。 : - )

我希望有所帮助。 : - )

-Jared

11年8月28日

@llango J

好的,好的,这是整个代码,欢迎您复制并粘贴它,看看是否可以让它工作: - )

对于初学者,您需要像这样调用相机:

startActivityForResult(CameraIntent, TAKE_PICTURE);

然后,结果你会想要这个:

@Override
public void onActivityResult(int requestCode, int resultCode, Intent intent) {
    if (resultCode == Activity.RESULT_OK) {
        switch (requestCode) {
        case TAKE_PICTURE:
            Uri u = intent.getData();
            //Toast.makeText(getApplicationContext(), u.getPath(), Toast.LENGTH_SHORT).show();
            WriteFiles(u);
            break;

        }
    }
}

然后主要的魔法发生在这个函数中:

private void WriteFiles(Uri myNewPic) {
    String TempFilePath;
    String TempPath;

    File directory = new File("/sdcard/" + getString(R.string.app_name));

    if (!directory.exists()) {
        directory.mkdirs();
    }

    String TempPictureFile;
    try {
        TempPictureFile = myNewPic.getLastPathSegment() + ".jpg";
        TempFilePath = directory.getPath() + "/" + TempPictureFile;
        FileOutputStream myOutStream = new FileOutputStream(TempFilePath);
        InputStream myInStream = getContentResolver().openInputStream(myNewPic);
        FileIO myFileIO = new FileIO();
        myFileIO.copy(myInStream, myOutStream);
        //now delete the file after copying
        getContentResolver().delete(myNewPic, null, null);
        TempFilePaths.add(TempFilePath);
    } catch (Exception e) {
        Toast.makeText(getApplicationContext(), getString(R.string.ErrorOpeningFileOutput), Toast.LENGTH_SHORT).show();
    }
}

我认为这是使用原生相机应用程序保存所需的所有部分。但是,我必须提醒你,我最终放弃了这个代码,因为我需要比这提供的更多的灵活性和可靠性(我想如果你在很多不同的手机上使用这个应用程序,我可以看到它出于某种原因出现问题例如,由于三星定制他们的用户界面,我遇到了一大堆问题,因此在所有手机上的功能都很好,但三星的功能却很好。:-P)。所以我最后创建了一个自己的相机应用程序,但我确信根据你的应用程序,这可能对你很好。无论哪种方式......祝你好运! : - )

好的,这是我使用的FileIO类,抱歉我最初没有包含这个:

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;

import android.util.Log;

public class FileIO {
private static final int BUFFER_SIZE = 1024 * 2;

public FileIO() {
    // Utility class.
}

public int copy(InputStream input, OutputStream output) throws Exception, IOException {
    byte[] buffer = new byte[BUFFER_SIZE];

    BufferedInputStream in = new BufferedInputStream(input, BUFFER_SIZE);
    BufferedOutputStream out = new BufferedOutputStream(output, BUFFER_SIZE);
    int count = 0, n = 0;
    try {
        while ((n = in.read(buffer, 0, BUFFER_SIZE)) != -1) {
            out.write(buffer, 0, n);
            count += n;
        }
        out.flush();
    } finally {
        try {
            out.close();
        } catch (IOException e) {
            Log.e(e.getMessage(), null);
        }
        try {
            in.close();
        } catch (IOException e) {
            Log.e(e.getMessage(), null);
        }
    }
    return count;
}

}