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);});}

流程

  1. 以 File 形式存儲拍攝的照片: 存儲拍攝后的照片到 getExternalCacheDir() 指定的應用關聯緩存目錄;(此時只有 output_image.jpg 這個文件名還沒有與之對應的照片)
  2. 將 File 對象轉為 Uri 對象: 7.0 版本后使用封裝過的 Uri 來替換原來標識真實路徑的 Uri,增強安全性;
  3. 將照片的輸出地址與 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,想要讀取視頻只需要將MediaStore.Images改為MediaStore.Video即可:

    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); // 根據圖片路徑顯示圖片}

進一步解析 documentcontent 類型初步解析得到的的 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();}}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/443696.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/443696.shtml
英文地址,請注明出處:http://en.pswp.cn/news/443696.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Android開發(4) | 系統權限、MediaPlayer類 和 VideoView類 的應用——播放多媒體文件

文章目錄MediaPlayer類播放音頻的實例VideoView類播放視頻的實例MediaPlayer類 對多種格式的音頻文件提供了全面的控制方法&#xff1a; 如何獲得MediaPlayer實例&#xff1f; 通過構造函數&#xff1a; MediaPlayer mp new MediaPlayer();調用 MediaPlayer.create() 方法&…

Android入門(15)| 網絡

文章目錄WebViewHTTP使用HttpURLConnection使用OkHttp封裝網絡操作封裝HttpURLConnection封裝OkHttpWebView WebView 可以在 應用程序中&#xff08;而不是瀏覽器&#xff09; 展示一些網頁。 布局文件 web_layout.xml&#xff1a; <LinearLayoutxmlns:android"http…

Java-單例模式

單例模式相信大家都不陌生&#xff0c;在JAVAEE應用中&#xff0c;單例模式是一種應用非常廣泛的設計模式&#xff0c;應用中許多組件都只需要單個實例&#xff0c;下面介紹單例模式。 使用單例模式的優點&#xff1a; 1.減少創建JAVA實例所帶來的系統開銷。 2.便于系統跟蹤單…

Android入門(16)| 服務

文章目錄概念Android 多線程繼承 Thread繼承 Runable 接口匿名類異步消息處理AsyncTask使用服務框架啟動/停止服務綁定/解綁服務服務的生命周期前臺服務IntentService完整版下載示例下載過程的回調接口&#xff1a;DownloadListener繼承 AsyncTask 實現下載功能&#xff1a;Dow…

2020德勤面試開始了嗎_2020國考面試開始,近期面試公告匯總,附結構化小組面試流程...

2020年國家公務員考試面試環節逐步恢復考試&#xff0c;各個招錄部門已經發布面試考察公告&#xff0c;對于進入面試環節的國考考生來說&#xff0c;有必要了解近期國考面試的招錄動態&#xff0c;提前做好面試準備。2020國考國家統計局機關面試面試確認&#xff1a;請進入面試…

項目積壓需求項目計劃_需求變更頻繁,項目經理如何做好需求管理?

項目實施過程中&#xff0c;項目經理常常面臨一個重大挑戰——需求變更。需求變更無處不在&#xff0c;市場條件變化、新業務出現、戰略目標調整、客戶需求修改、資源限制等&#xff0c;都會造成需求變更。需求變更會影響項目的時間、成本和質量&#xff0c;對整個項目和團隊成…

Android | Sensor.TYPE_ORIENTATION被廢棄后的解決辦法

文章目錄概述getOrientation 方法根據 旋轉矩陣R 獲取 設備旋轉弧度getRotationMatrix 方法根據 地磁場、加速度傳感器對象 獲取 旋轉矩陣R代碼參考資料概述 Sensor.TYPE_ORIENTATION 常數在 API 8 中已棄用&#xff0c;官方推薦使用 SensorManager.getOrientation() 替代。關…

【JAVA 開發小問題】 | String操作合集

文章目錄截取特定兩個字符之間的字符串截取特定兩個字符之間的字符串 利用正則表達式&#xff0c;圖片來源

uniapp 刷新后數據都沒有了_環境溫度傳感器都沒有連接,竟然還會有數據?

福田歐曼GTL(福康發動機、康明斯2880系統)匹配ECoffit尿素泵●故障現象&#xff1a;OBD故障燈點亮&#xff0c;不燒尿素&#xff0c;油耗高&#xff0c;動力不足●故障碼&#xff1a;●維修分析&#xff1a;①故障指出加熱器問題&#xff0c;摸下尿素箱溫度&#xff0c;發現燙手…

Android | 再探 RecyclerView 之名詞解析

文章目錄Adapter、ViewHolderchild viewLayoutManagerRecyclerScrapDirtyIndexPositionlayout position 和 adapter position四級緩存瀏覽本文前推薦先閱讀 Android入門&#xff08;九&#xff09;| 滾動控件 ListView 與 RecyclerView Adapter、ViewHolder Adapter: A subcla…

Linux學習:第一章-Linux簡介

一 UNIX發展史1 1965年&#xff0c;美國麻省理工學院&#xff08;MIT&#xff09;、通用電氣公司&#xff08;GE&#xff09;及AT&T的貝爾實驗室聯合開發Multics工程計劃&#xff0c;其目標是開發一種交互式的具有多道程序處理能力的分時操作系統&#xff0c;但因Multics追…

尼爾機器人技能快捷鍵_《尼爾機械紀元》連招操作技巧

《尼爾機械紀元》中的每個角色都可以裝備一個輕武器和一個重武器&#xff0c;技能招式也很豐富&#xff0c;下面為大家帶來了《尼爾機械紀元》連招操作技巧&#xff0c;希望對你們有所幫助。連擊技能展示視頻視頻原址&#xff1a;點擊進入基本沒有什么太難的連段&#xff0c;只…

保姆級教學!Xcode 配置 OpenGL 環境

文章目錄GLFW獲取 GLFWGLAD獲取 GLAD在 Xcode 中配置下載好的 GLFW 和 GLAD配置流程檢測是否配置成功無關配置的題外話——Xcode 下安全的刪除移動操作GLFW Graphics Library Framework&#xff08;圖形庫框架&#xff09;&#xff0c;可以讓我們通過其封裝好的 通用API 來正確…

Android入門(17)| 百度提供的 Android定位SDK

文章目錄配置百度提供的 Android定位SDK用于發布的 SHA1用于測試的 SHA1使用百度定位實例配置百度提供的 Android定位SDK 詳情參見官方文檔&#xff0c;這里僅對獲取 SHA1 做詳細介紹&#xff1a; 用于發布的 SHA1 用于測試的 SHA1 使用百度定位實例 public class LocationAc…

ios 不被遮擋 陰影_為何你沒見到日環食?你不知道的天象常識原來還有這么多 | 返樸...

關注風云之聲提升思維層次導讀說好的日環食呢&#xff0c;為什么上周很多人只等到了日偏食?日食月食的時間和種類是怎么預測的?你真的弄懂了各種日食和月食的成因嗎&#xff1f;你了解它們有什么區別和聯系&#xff0c;又遵循什么樣的時間規律嗎? 日食和月食發生的頻率一樣嗎…

初識貝塞爾(bezier)曲線

文章目錄資料援引貝塞爾曲線的用途一階貝塞爾&#xff08;bezier&#xff09;曲線二階貝塞爾&#xff08;bezier&#xff09;曲線三階貝塞爾&#xff08;bezier&#xff09;曲線高階貝塞爾&#xff08;bezier&#xff09;曲線三階貝塞爾曲線求插值&#xff08;Slerp&#xff09…

python代碼測試 vim_用 Hypothesis 快速測試你的 Python 代碼

點擊上方“Python編程時光”&#xff0c;選擇“加為星標”第一時間關注Python技術干貨&#xff01;介紹無論你使用哪種編程語言或框架&#xff0c;測試都非常重要。Hypothesis是 Python 的一個高級測試庫。它允許編寫測試用例時參數化&#xff0c;然后生成使測試失敗的簡單易懂…

Mac 下 CMake 的配置與使用

文章目錄安裝與配置編譯單個源文件編譯前的準備開始編譯編譯多個源文件多個源文件在同一目錄下多個源文件在不同目錄下math 目錄下的 CMakeLists.txt根目錄的 CMakeLists.txtoption 選項導入外部庫本地導入&#xff08;find_package&#xff09;外部導入&#xff08;FetchConte…

五軸編程_沙井萬豐數控數控編程五軸編程那個軟件好用

沙井萬豐數控數控編程五軸編程那個軟件好用設計需要掌握很高很全面的知識和技能&#xff0c;模具做的好&#xff0c;產品質量好&#xff0c;模具結構合理&#xff0c;生產效率高&#xff0c;工廠效益好。正因如此&#xff0c;模具技術工在外打工的工資都非常的高。少則每月幾千…

Linux學習:第二章-Linux安裝

一虛擬機使用 VMware主要特點&#xff1a; 1、不需要分區或重新開機就能在同一臺PC上使用兩種以上的操作系統 2、本機系統可以與虛擬機系統網絡通信 3、可以設定并且隨時修改虛擬機操作系統的硬件環境 二安裝方式 圖形安裝&#xff1a;直接回車 字符安裝&#xff1a;linux tex…