Обновление (2020)
Google добавил новый ActivityResultRegistry
API, который «позволяет вам обрабатывать потоки « startActivityForResult()
+ » onActivityResult()
и« requestPermissions()
+ » onRequestPermissionsResult()
без переопределения методов в вашей деятельности или фрагменте, обеспечивает повышенную безопасность типов ActivityResultContract
и обеспечивает ловушки для тестирования этих потоков» - источник .
API был добавлен в androidx.activity 1.2.0-alpha02 и androidx.fragment 1.3.0-alpha02 .
Теперь вы можете сделать что-то вроде:
val takePicture = registerForActivityResult(ActivityResultContracts.TakePicture()) { success: Boolean ->
if (success) {
// The image was saved into the given Uri -> do something with it
}
}
val imageUri: Uri = ...
button.setOnClickListener {
takePicture.launch(imageUri)
}
Посмотрите документацию, чтобы узнать, как использовать новый API результатов действий: https://developer.android.com/training/basics/intents/result#kotlin
Есть много встроенных ActivityResultContracts, которые позволяют вам делать разные вещи, такие как выбор контактов, запрос разрешений, фотографировать или снимать видео. Вы, вероятно, заинтересованы в ActivityResultContracts.TakePicture, показанном выше.
Обратите внимание, что androidx.fragment 1.3.0-alpha04 не поддерживает API startActivityForResult()
+ onActivityResult()
и requestPermissions()
+ onRequestPermissionsResult()
для Fragment. Следовательно, кажется, что ActivityResultContracts
это новый способ делать вещи отныне.
Оригинальный ответ (2015)
Мне понадобилось несколько часов, чтобы заставить это работать. Код - это почти копия-вставка с developer.android.com , с небольшим отличием.
Запросите это разрешение на AndroidManifest.xml
:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
На вашем Activity
, начнем с определения этого:
static final int REQUEST_IMAGE_CAPTURE = 1;
private Bitmap mImageBitmap;
private String mCurrentPhotoPath;
private ImageView mImageView;
Затем запустите это Intent
в onClick
:
Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (cameraIntent.resolveActivity(getPackageManager()) != null) {
// Create the File where the photo should go
File photoFile = null;
try {
photoFile = createImageFile();
} catch (IOException ex) {
// Error occurred while creating the File
Log.i(TAG, "IOException");
}
// Continue only if the File was successfully created
if (photoFile != null) {
cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(photoFile));
startActivityForResult(cameraIntent, REQUEST_IMAGE_CAPTURE);
}
}
Добавьте следующий метод поддержки:
private File createImageFile() throws IOException {
// Create an image file name
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
String imageFileName = "JPEG_" + timeStamp + "_";
File storageDir = Environment.getExternalStoragePublicDirectory(
Environment.DIRECTORY_PICTURES);
File image = File.createTempFile(
imageFileName, // prefix
".jpg", // suffix
storageDir // directory
);
// Save a file: path for use with ACTION_VIEW intents
mCurrentPhotoPath = "file:" + image.getAbsolutePath();
return image;
}
Тогда получите результат:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) {
try {
mImageBitmap = MediaStore.Images.Media.getBitmap(this.getContentResolver(), Uri.parse(mCurrentPhotoPath));
mImageView.setImageBitmap(mImageBitmap);
} catch (IOException e) {
e.printStackTrace();
}
}
}
То, что заставило это работать, - то MediaStore.Images.Media.getBitmap(this.getContentResolver(), Uri.parse(mCurrentPhotoPath))
, что отличается от кода от developer.android.com . Оригинальный код дал мне FileNotFoundException
.