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.mzph.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…

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;发现烫手…

保姆级教学!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…

keil3如何放大字体_国潮海报不会做?送你国风字体+图案笔刷+PSD素材+包装样机...

有很多朋友都问带鱼&#xff0c;国潮风的海报到底应该怎么做呢&#xff1f;首先你要知道什么是国潮风&#xff1a;国潮风就是现代文化和古代文化的碰撞&#xff0c;是年轻人的态度&#xff01;那么应该如何构图如何设计呢&#xff1f;如何配色如何搭配字体呢&#xff1f;这些方…

Google 开源项目风格指南学习笔记——C++篇

文章目录前言0. 缩写名词解释1. 头文件1.1. Self-contained 头文件1.2. 头文件保护1.3. 前置声明1.4 内联函数1.5. #include 的路径及顺序2. 作用域2.1. 命名空间2.2. 非成员函数、静态成员函数和全局函数2.3. 局部变量2.4. 静态和全局变量3. 类3.1. 构造函数的职责3.2. 隐式类…

hiveserver2启动不起来_给爱车配个充电宝,70迈汽车应急启动电源,让你远离搭电小广告...

说到汽车应急启动其实我有切身的痛&#xff0c;在哈尔滨零下35的严冬&#xff0c;晚上带着女神吃完饭&#xff0c;高高兴兴地吃完以后一上车&#xff0c;发现电瓶被冻没电了&#xff0c;天知道当时有多尴尬。马上叫了保险公司过来给搭电&#xff0c;结果在饭店从晚上8点一直等到…

Windows 下 VS 配置 OpenGL 环境

文章目录前言获取 GLFW打开 VS前言 感谢B站同学搬运YouTube上的教学视频。 获取 GLFW 从官网上下载GLFW macOS下64位二进制文件 打开 VS 新建解决方案 OpenGL test&#xff0c;并在解决方案中新建文件夹 Dependencies&#xff1a; 从下载好的 glfw 文件夹中找到最新版链接…