Оновлення (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 зменшує значущість startActivityForResult()
+ onActivityResult()
і requestPermissions()
+ onRequestPermissionsResult()
API , на фрагменті. Отже, здається, що 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
.