본문 바로가기
Android/Android 기초

[Android] 안드로이드 저장소(3) - 카메라/갤러리 저장하기 예제(Q Scoped Storage)

by 태크민 2024. 1. 7.

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

https://velog.io/@bonimddal2/%EC%95%88%EB%93%9C%EB%A1%9C%EC%9D%B4%EB%93%9C-%EC%95%88%EB%93%9C%EB%A1%9C%EC%9D%B4%EB%93%9C-QAPI-29%EC%9D%B4%EC%83%81%EC%97%90%EC%84%9C-%EC%9D%B4%EB%AF%B8%EC%A7%80-%ED%8C%8C%EC%9D%BC-%EB%8B%A4%EB%A3%A8%EA%B8%B0

 

[안드로이드] 안드로이드 Q(API 29)이상에서 이미지 파일 다루기

안드로이드 빌드 버전 Q (안드로이드 10, API 29) 이상부터 Scoped Storage가 적용되어, 갤러리에 이미지를 저장하거나, 기기에 있는 이미지 파일을 서버에 업로드하기가 어려워졌다. > 내부 vs 외부 저

velog.io