日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android开发(3) | 权限和内容提供器的应用——调用相机和相册

發布時間:2023/12/13 Android 27 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android开发(3) | 权限和内容提供器的应用——调用相机和相册 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

文章目錄

  • 拍照并保存到 ImageView 控件
    • 布局文件 notice_layout.xml
    • 按鈕 button_takePhoto 的點擊操作
    • 隱式 Intent 啟動后的回調
    • AndroidManifest.xml
  • 從相冊選取照片并在 ImageView 控件中顯示
    • 布局文件 notice_layout.xml
    • 按鈕 button_takePhoto 的點擊操作
    • 自定義打開相冊的方法 openAlbum
    • 隱式 Intent 啟動后的回調


拍照并保存到 ImageView 控件

布局文件 notice_layout.xml

按鈕 button_takePhoto 的點擊操作

public static final int TAKE_PHOTO = 1;private ImageView picture;private Uri imageUri;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.notice_layout);picture = findViewById(R.id.picture);Button button_takePhoto = findViewById(R.id.button_takePhoto);button_takePhoto.setOnClickListener(v->{// 存儲拍攝后的照片到 getExternalCacheDir() 指定的應用關聯緩存目錄File outputImage = new File(getExternalCacheDir(), "output_image.jpg");try {// 除首次拍照外,都需要刪除原有存在的舊照片if(outputImage.exists()){outputImage.delete();}// 再創建新文件outputImage.createNewFile();} catch (IOException e) {e.printStackTrace();}// 7.0 版本后,直接使用標識本地真實路徑的Uri會拋出 FileUriExposedException 異常if(Build.VERSION.SDK_INT >= 24){// getUriForFile三個參數:Context對象、任意唯一字符串、File對象// 參數二必須和AndroidManifest.xml中provider標簽的authorities屬性一致// 作用是將File對象轉為封裝過的Uri對象,提高安全性imageUri = FileProvider.getUriForFile(this,"com.example.activitytest.Activity.fileProvider", outputImage);}else{// fromFile將File對象轉為標識圖片本地真實路徑的Uri對象imageUri = Uri.fromFile(outputImage);}// 指定開啟系統相機的ActionIntent intent = new Intent("android.media.action.IMAGE_CAPTURE");// 指定圖片的輸出地址為之前創建的Uri對象imageUriintent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);// 隱式Intent,startActivityForResult之后回調onActivityResultstartActivityForResult(intent, TAKE_PHOTO);});}

流程

  • 以 File 形式存儲拍攝的照片: 存儲拍攝后的照片到 getExternalCacheDir() 指定的應用關聯緩存目錄;(此時只有 output_image.jpg 這個文件名還沒有與之對應的照片)
  • 將 File 對象轉為 Uri 對象: 7.0 版本后使用封裝過的 Uri 來替換原來標識真實路徑的 Uri,增強安全性;
  • 將照片的輸出地址與 Uri 對象綁定: 此時才完成了 通過 Intent 跳轉到相機、通過 Uri對象 將拍攝好的照片與文件名 output_image.jpg 綁定 的代碼邏輯。
  • 為什么使用應用關聯緩存目錄存放圖片?

    首先明確該目錄的路徑是 /sdcard/Android/data/<packge name>/cache。

    從 Android 6.0 開始,讀寫 SD 卡被列為危險權限,如果將圖片放在 SD 卡的任何其他目錄,都要進行運行時權限處理,而使用應用關聯緩存目錄無需進行。

    隱式 Intent 啟動后的回調

    // startActivityForResult之后回調onActivityResult@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);switch (requestCode) {case TAKE_PHOTO:if (resultCode == RESULT_OK) {try {// 將output_image.jpg解析成Bitmap對象Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));// 設置到ImageView中picture.setImageBitmap(bitmap);} catch (FileNotFoundException e) {e.printStackTrace();}}break;default:break;}}

    AndroidManifest.xml

    為了兼容 4.4及之前 的系統,需要聲明訪問SD卡的權限:

    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

    provider 標簽:

    • name 屬性的值是固定的
    • authorities 屬性的值必須和 FileProvider.getUriForFile() 方法中參數二一致
    • <meta-adta> 中的 resource 屬性的值是我們自創的文件


    從相冊選取照片并在 ImageView 控件中顯示

    布局文件 notice_layout.xml

    按鈕 button_takePhoto 的點擊操作

    public static final int CHOOSE_PHOTO = 2;protected void onCreate(Bundle savedInstanceState) {Button button_album = findViewById(R.id.button_album);button_album.setOnClickListener(v->{// WRITE_EXTERNAL_STORAGE:對SD卡讀和寫的權限// 相等說明用戶已授權,不等說明未授權if(ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){ActivityCompat.requestPermissions(this,new String[] { Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);}else {openAlbum();}});}

    請求授權 requestPermissions 的回調 onRequestPermissionsResult:

    // ActivityCompat.requestPermissions結束后回調@Overridepublic void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults){switch (requestCode) {case 1:if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {openAlbum();}break;default:}if(!ActivityCompat.shouldShowRequestPermissionRationale(this,Manifest.permission.WRITE_EXTERNAL_STORAGE)){AlertDialog.Builder dialog = new AlertDialog.Builder(this);dialog.setTitle("圖庫權限不可用").setMessage("請在-應用設置-權限中,允許APP使用圖庫權限。").setCancelable(false).setPositiveButton("立即設置", (dialog1, which) -> goToAppSetting()).setNegativeButton("取消", (dialog2, which) -> dialog2.dismiss()).show();}}

    用戶未授權卻想授權時跳轉到權限設置界面:

    // 跳轉到權限設置界面private void goToAppSetting() {Intent intent = new Intent();intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);Uri uri = Uri.fromParts("package", getPackageName(), null);intent.setData(uri);startActivity(intent);}

    自定義打開相冊的方法 openAlbum

    // 打開相冊private void openAlbum(){// 獲取內容,具體調用哪那個程序由type屬性決定Intent intent = new Intent("android.intent.action.GET_CONTENT");// 設置type屬性,調用圖庫intent.setType("image/*");// 啟動隱式Intent,跳轉到相冊startActivityForResult(intent, CHOOSE_PHOTO);}

    隱式 Intent 啟動后的回調

    // startActivityForResult之后回調onActivityResult@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);switch (requestCode) {case CHOOSE_PHOTO:// 處理圖片if(resultCode == RESULT_OK){// 4.4及以上系統會對Uri進行封裝,需要進一步解析才能得到真實Uriif(Build.VERSION.SDK_INT >= 19){handleImageOnKitKat(data);}// 4.4以下系統可以直接獲得真實Urielse{handleImageBeforeKitKat(data);}}default:break;}}

    4.4以下系統可以直接獲得 真實Uri

    private void handleImageBeforeKitKat(Intent data){Uri uri = data.getData();String imagePath = getImagePath(uri, null);displayImage(imagePath);}

    4.4及以上系統需要進一步解析 封裝的Uri 才能得到 真實Uri

    private void handleImageOnKitKat(Intent data) {String imagePath = null;Uri uri = data.getData();// 如果是document類型的Uriif(DocumentsContract.isDocumentUri(this, uri)){// 則通過document id處理String docId = DocumentsContract.getDocumentId(uri);// 如果authority是media格式,需要分割字符串得到真正的數字idif("com.android.providers.media.documents".equals(uri.getAuthority())){// 根據":"分割docIdString id = docId.split(":")[1]; // 解析出數字格式的idString selection = MediaStore.Images.Media._ID + "=" + id;// EXTERNAL_CONTENT_URI:“主”外部存儲卷的樣式URI。imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);}else if("com.android.providers.downloads.documents".equals(uri.getAuthority())){Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads" +"/public_downloads"), Long.valueOf(docId));imagePath = getImagePath(contentUri, null);}// 如果是content類型的Uri} else if("content".equalsIgnoreCase(uri.getScheme())){// 則使用普通方式處理imagePath = getImagePath(uri, null);// 如果是file類型的Uri} else if("file".equalsIgnoreCase(uri.getScheme())){// 直接獲取圖片路徑即可imagePath = uri.getPath();}displayImage(imagePath); // 根據圖片路徑顯示圖片}

    進一步解析 document、content 類型初步解析得到的的 Uri:

    // 進一步解析真實Uriprivate String getImagePath(Uri uri, String selection){String path = null;Cursor cursor = getContentResolver().query(uri, null, selection,null, null);if(cursor != null){if(cursor.moveToFirst()){path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));}cursor.close();}return path;}

    在 ImageView 中顯示圖片:

    // 在ImageView中顯示圖片private void displayImage(String imagePath){if(imagePath != null){Bitmap bitmap = BitmapFactory.decodeFile(imagePath);picture.setImageBitmap(bitmap);}else{Toast.makeText(this, "failed to get image", Toast.LENGTH_LONG).show();}}

    總結

    以上是生活随笔為你收集整理的Android开发(3) | 权限和内容提供器的应用——调用相机和相册的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。