Android 11 타겟팅을 위한 권한 설정 (Manifest)
앱이 Android 11을 타겟팅하는 경우 WRITE_EXTERNAL_STORAGE 권한 및 WRITE_MEDIA_STORAGE 독점 권한은 더 이상 추가 액세스를 제공하지 않는다.
따라서, MediaStore를통한 파일 접근 방식이 필요하다.
- MANAGE_EXTERNAL_STORAGE 권한 선언
- 앱에 관해 모든 파일 관리 허용 옵션을 사용 설정할 수 있는 시스템 설정 페이지로 사용자를 안내함
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission
android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="28" />
<uses-permission
android:name="android.permission.READ_EXTERNAL_STORAGE"
android:maxSdkVersion="28" />
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />
MANAGE_EXTERNAL_STORAGE 권한은 다음을 허용한다.
- 공유 저장소 내의 모든 파일에 관한 읽기 액세스 및 쓰기 액세스
- MediaStore.Files 테이블의 콘텐츠에 관한 액세스
--> 저장소에 읽기/쓰기를 해야하는 경우 따로 권한이 추가하여 필요하다는 뜻으로 파악됨.
다른 앱과 파일을 공유하는 방법 (Manifest)
다른 앱과 파일을 공유하기 위해서는 FileProvider를 이용해서 해당 파일에 접근할 수 있는 Uri를 만들고, 파일을 공유받을 대상 앱에 임시로 Uri 접근(읽기 및 쓰기) 권한을 허용하는 방법을 사용해야 한다.
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="com.appgrider.test_android.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-path
name="external"
path="." />
<cache-path
name="cache"
path="." />
<external-cache-path
name="external_cache"
path="." />
<files-path
name="files"
path="." />
<external-files-path name="external_files" path="Android/data/com.appgrider.test_android/files/Pictures" />
</paths>
카메라로 찍은 사진 갤러리에 저장하기
퍼미션 설정
public void checkPermission(){
String[] permission_list = {
Manifest.permission.CAMERA,
Manifest.permission.WRITE_EXTERNAL_STORAGE,
Manifest.permission.READ_EXTERNAL_STORAGE
};
//현재 안드로이드 버전이 6.0미만이면 메서드를 종료한다.
if(Build.VERSION.SDK_INT < Build.VERSION_CODES.M)
return;
for(String permission : permission_list){
//권한 허용 여부를 확인한다.
int chk = checkCallingOrSelfPermission(permission);
if(chk == PackageManager.PERMISSION_DENIED){
//권한 허용을여부를 확인하는 창을 띄운다
requestPermissions(permission_list,0);
}
}
}
카메라로 열기
private void camera_open_intent() {
Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
File photoFile = null;
try {
photoFile = createImageFile();
} catch (IOException ex) {
// Error occurred while creating the File
Log.e("Woongs",ex.getMessage().toString());
}
// Continue only if the File was successfully created
if (photoFile != null) {
Uri providerURI = FileProvider.getUriForFile( this , "com.appgrider.test_android.fileprovider", photoFile);
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, providerURI);
startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
}
}
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) {
Uri test = Uri.fromFile(new File(currentPhotoPath));
saveFile(test);
}
}
파일 생성
// 파일 생성
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 = getExternalFilesDir(Environment.DIRECTORY_PICTURES);
// 참고: getExternalFilesDir() 또는 getFilesDir()에서 제공한 디렉터리에 저장한 파일은 사용자가 앱을 제거할 때 삭제됩니다.
File image = File.createTempFile(
imageFileName, /* prefix */
".jpg", /* suffix */
storageDir /* directory */
);
// Save a file: path for use with ACTION_VIEW intents
currentPhotoPath = image.getAbsolutePath();
return image;
}
카메라로 찍은사진 갤러리에 저장
// 파일 저장
private void saveFile(Uri image_uri) {
ContentValues values = new ContentValues();
String fileName = "woongs"+System.currentTimeMillis()+".png";
values.put(MediaStore.Images.Media.DISPLAY_NAME,fileName);
values.put(MediaStore.Images.Media.MIME_TYPE, "image/*");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
//IS_PENDING을 1로 설정해놓으면, 현재 파일을 업데이트 전까지 외부에서 접근하지 못하도록 할 수 있다.
values.put(MediaStore.Images.Media.IS_PENDING, 1);
}
ContentResolver contentResolver = getContentResolver();
Uri item = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
try {
ParcelFileDescriptor pdf = contentResolver.openFileDescriptor(item, "w", null);
if (pdf == null) {
Log.d("Woongs", "null");
} else {
byte[] inputData = getBytes(image_uri);
FileOutputStream fos = new FileOutputStream(pdf.getFileDescriptor());
fos.write(inputData);
fos.close();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
values.clear();
// 파일 저장이 완료되었으니, IS_PENDING을 다시 0으로 설정한다.
values.put(MediaStore.Images.Media.IS_PENDING, 0);
// 파일을 업데이트하면, 파일이 보인다.
contentResolver.update(item, values, null, null);
}
// 갱신
galleryAddPic(fileName);
}
} catch (FileNotFoundException e) {
e.printStackTrace();
Log.d("Woongs", "FileNotFoundException : "+e.getLocalizedMessage());
} catch (Exception e) {
Log.d("Woongs", "FileOutputStream = : " + e.getMessage());
}
}
그밖에 유틸
// Uri to ByteArr
public byte[] getBytes(Uri image_uri) throws IOException {
InputStream iStream = getContentResolver().openInputStream(image_uri);
ByteArrayOutputStream byteBuffer = new ByteArrayOutputStream();
int bufferSize = 1024; // 버퍼 크기
byte[] buffer = new byte[bufferSize]; // 버퍼 배열
int len = 0;
// InputStream에서 읽어올 게 없을 때까지 바이트 배열에 쓴다.
while ((len = iStream.read(buffer)) != -1)
byteBuffer.write(buffer, 0, len);
return byteBuffer.toByteArray();
}
갤러리 갱신
private void galleryAddPic(String Image_Path) {
Log.d("Woongs","갱신 : "+Image_Path);
// 이전 사용 방식
/*Intent mediaScanIntent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
File f = new File(Image_Path);
Uri contentUri = Uri.fromFile(f);
mediaScanIntent.setData(contentUri);
this.context.sendBroadcast(mediaScanIntent);*/
File file = new File(Image_Path);
MediaScannerConnection.scanFile(context,
new String[]{file.toString()},
null, null);
}
갤러리 호출
private void gallery_open_intent(){
Intent intent = new Intent(Intent.ACTION_PICK,MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(intent,REQUEST_GALLERY_IMAGE);
}
출처
https://machine-woong.tistory.com/453
Android 카메라 / 갤러리 저장하기 ( Q Scoped Storage)
Manifesats res - xml - file_paths.xml data/ 본인의 패키지명 / files / 디렉터리명 // 퍼미션은 있다고 가정하겠습니다. // 카메라 열기 private void camera_open_intent() { Intent takePictureIntent = new Intent(MediaStore.ACTION_IM
machine-woong.tistory.com
[안드로이드] 안드로이드 Q(API 29)이상에서 이미지 파일 다루기
안드로이드 빌드 버전 Q (안드로이드 10, API 29) 이상부터 Scoped Storage가 적용되어, 갤러리에 이미지를 저장하거나, 기기에 있는 이미지 파일을 서버에 업로드하기가 어려워졌다. > 내부 vs 외부 저
velog.io
'Android > Android 기초' 카테고리의 다른 글
[Android] 메모리 누수 & 분석 (0) | 2024.06.03 |
---|---|
[Android] 빌드 변형 구성 (ProductFlavors) 한 개의 프로젝트로 여러개의 앱 만들기 (0) | 2024.02.18 |
[Android] 안드로이드 저장소(2) - Scoped Storage (1) | 2024.01.07 |
[Android] 안드로이드 저장소(1) - Legacy Storage (1) | 2024.01.07 |
[Android] 프래그먼트(Fragment) 간 통신 방법 4가지 (0) | 2023.09.06 |