安卓相册设置_安卓11原生相机

安卓相册设置_安卓11原生相机前言适配前台程序员必不可少的工作之一,且可能要花大量的时间精力。何为前台程序员,是面向用户的一端,包括前端、移动端、PC等等。何为适配,适配就是当我们的开发环境、运行环境等发生变化的时候,程序依然能稳健运行。而适配中最难为程序员的就是Android了,除了开发环境、运行环境等因素之外,因为Android开源的原因,还要适配各大厂商。。而适配条件之多,经常让Android程序员为之头疼。来看看相机、相册相关的适配历程:Android6权限适配Android7文件适配Android

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

前言

适配前台程序员必不可少的工作之一,且可能要花大量的时间精力。

何为前台程序员,是面向用户的一端,包括前端、移动端、PC等等。

何为适配,适配就是当我们的开发环境运行环境等发生变化的时候,程序依然能稳健运行。

而适配中最难为程序员的就是Android了,除了开发环境、运行环境等因素之外,因为Android开源的原因,还要适配各大厂商。。

而适配条件之多,经常让Android程序员为之头疼。

来看看相机相册相关的适配历程:

  • Android 6 权限适配
  • Android 7 文件适配
  • Android 10/11 存储适配

ok,接下来以一个更换头像的小例子来讲解一下。

示例

在这里插入图片描述

点击头像,然后弹窗,给出不同的选项,执行不同的操作。

mBinding.llImg.setOnClickListener { 
   
    TakeImageDialog { 
   
        when (it) { 
   
            TakeImageDialog.ALBUM -> { 
    
                openAlbum()
            }
            TakeImageDialog.CAMERA -> { 
    
                checkPermission()
            }
        }
    }.show(supportFragmentManager, "TakeImageDialog")
}

定义后面会用到的一些参数变量

    //相机拍照保存的位置
    private lateinit var photoUri: Uri
    
    companion object { 
   
        private const val REQUEST_CODE_PERMISSIONS = 1000 //权限
        private const val REQUEST_CODE_ALBUM = 1001 //相册
        private const val REQUEST_CODE_CAMERA = 1002 //相机
    }

打开相册

选择图片

    private fun openAlbum() { 
   
        val intent = Intent()
        intent.type = "image/*"
        intent.action = "android.intent.action.GET_CONTENT"
        intent.addCategory("android.intent.category.OPENABLE")
        startActivityForResult(intent, REQUEST_CODE_ALBUM)
    }

固定写法,大差不差。

既然是startActivityForResult启动方式,来看看onActivityResult回调

回调

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { 
   
    super.onActivityResult(requestCode, resultCode, data)
    if (resultCode == RESULT_OK) { 
   
        when (requestCode) { 
   
            REQUEST_CODE_ALBUM -> { 
   
                doCrop(data?.data!!)
            }
            ...
        }
    }
}

requestCodeREQUEST_CODE_ALBUM的情况下:

doCrop(data?.data!!)

data?.data!!即是选择图片返回的Uri,可以直接使用,这里进行了下一步操作,剪裁

剪裁

    private fun doCrop(sourceUri: Uri) { 
   
        Intrinsics.checkParameterIsNotNull(sourceUri, "资源为空")
        UCrop.of(sourceUri, getDestinationUri())//当前资源,保存目标位置
            .withAspectRatio(1f, 1f)//宽高比
            .withMaxResultSize(500, 500)//宽高
            .start(this)
    }

为了方便,这里使用了一个三方库UCrop,使用简单方便。

getDestinationUri()是当前资源裁剪后保存的目标位置

    private fun getDestinationUri(): Uri { 
   
        val fileName = String.format("fr_crop_%s.jpg", System.currentTimeMillis())
        val cropFile = File(getExternalFilesDir(Environment.DIRECTORY_PICTURES), fileName)
        return Uri.fromFile(cropFile)
    }

UCrop的回调同样也在onActivityResult

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { 
   
    super.onActivityResult(requestCode, resultCode, data)
    if (resultCode == RESULT_OK) { 
   
        when (requestCode) { 
   
            REQUEST_CODE_ALBUM -> { 
   
                doCrop(data?.data!!)
            }
            UCrop.REQUEST_CROP -> { 
   
                val resultUri: Uri = UCrop.getOutput(data!!)!!
                val bitmap = BitmapFactory.decodeStream(contentResolver.openInputStream(resultUri))
                // todo
            }
            UCrop.RESULT_ERROR -> { 
   
                val error: Throwable = UCrop.getError(data!!)!!
                ToastUtil.show("图片剪裁失败" + error.message)
            }
        }
    }
}

UCrop.getOutput(data!!)!!,即是返回的Uri,可以直接操作,也可以转成bitmap

ok,到这里打开相册就介绍完了。

接下来看重点,打开相机。

author:yechaoa

打开相机

打开相机的流程就要稍微复杂一点了。

权限

第一步不是打开,而是先检查是否有相机权限,这个在某些手机上是必须的,比如华为。

  • 配置文件添加:
<uses-permission android:name="android.permission.CAMERA" />
  • 代码:
    private fun checkPermission() { 
   
        if (ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED) { 
   
            openCamera()
        } else { 
   
            ActivityCompat.requestPermissions(this, arrayOf(Manifest.permission.CAMERA), REQUEST_CODE_PERMISSIONS)
        }
    }
  • 回调:
    override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<out String>, grantResults: IntArray) { 
   
        super.onRequestPermissionsResult(requestCode, permissions, grantResults)
        if (requestCode == REQUEST_CODE_PERMISSIONS) { 
   
            if (grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED) { 
   
                openCamera()
            } else { 
   
                ToastUtil.show("拒绝会导致无法使用相机")
            }
        }
    }

openCamera方法就是打开相机了。

打开前适配

    private fun openCamera() { 
   
        val intent = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
        photoUri = getDestinationUri()
        photoUri = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { 
   
            //适配Android 7.0文件权限,通过FileProvider创建一个content类型的Uri
            FileProvider.getUriForFile(this, "$packageName.fileProvider", File(photoUri.path!!))
        } else { 
   
            getDestinationUri()
        }
        //android11以后强制分区存储,外部资源无法访问,所以添加一个输出保存位置,然后取值操作
        intent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri)
        startActivityForResult(intent, REQUEST_CODE_CAMERA)
    }
  • 适配一:
FileProvider.getUriForFile(this, "$packageName.fileProvider", File(photoUri.path!!))

7.0以上,使用fileProvider的方式共享文件。

  • 适配二:
intent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri)

android 11以后强制分区存储,外部资源无法访问,所以添加一个输出保存位置photoUri,然后取值操作

回调

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { 
   
    super.onActivityResult(requestCode, resultCode, data)
    if (resultCode == RESULT_OK) { 
   
        when (requestCode) { 
   
            REQUEST_CODE_ALBUM -> { 
   
                doCrop(data?.data!!)
            }
            REQUEST_CODE_CAMERA -> { 
   
                //从保存的位置取值
                doCrop(photoUri)
            }
            UCrop.REQUEST_CROP -> { 
   
                val resultUri: Uri = UCrop.getOutput(data!!)!!
                val bitmap = BitmapFactory.decodeStream(contentResolver.openInputStream(resultUri))
                // todo
            }
            UCrop.RESULT_ERROR -> { 
   
                val error: Throwable = UCrop.getError(data!!)!!
                ToastUtil.show("图片剪裁失败" + error.message)
            }
        }
    }

这里注意,不是相册那样从data取值了,而是从我们保存的位置里取值。

后面剪裁跟相册都是一样的流程了。

总结

这个功能点最大的变动就是分区存储了,Android 10或许还能过度一下,但是Android 11以后就是强制执行分区存储了。

应用可以在不需要读写权限的情况下,访问自己的分区,执行读写操作,卸载之后分区文件也相应删除,所以就不能有把缓存文件放到竞品的文件夹下这种操作了,还是乖乖的吧。

在Android 10以下,还是要读写权限的,还是可以胡作非为的。

获取自己的分区地址:

getExternalFilesDir(Environment.DIRECTORY_PICTURES)

对应地址:

file:///storage/emulated/0/Android/data/包名/files/Pictures

file开头是沙盒文件,content开头是共享文件。

那假如我有访问其他文件的需求呢,比如相册、音乐,那还是需要读写权限的,且得通过MediaStore API来进行访问了,具体可以查看文档

最后

写作不易,如果对你有用,点个赞呗 ^ _ ^

Android 11开发手册

《Android 11 开发者手册》

参考

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/190541.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • labview如何旋转控件_labview怎么用

    labview如何旋转控件_labview怎么用龙源期刊网http://www.qikan.com.cn基于LabVIEW的旋转倒立摆系统设计作者:白富斌董君浩侯丽鹏来源:《现代商贸工业》2016年第09期摘要:以LabVIEW为平台,结合PID算法,对旋转倒立摆系统设计进行设计研究。在倒立摆旋转过程中,通过编码器将判断位置与角度的相应电信号反馈给上位机,上位机通过运行程序计算并输出信号进而来控制摆杆的的角度、位置,使倒立摆的摆杆不会下垂。关键…

    2022年8月18日
    5
  • javah命令详解「建议收藏」

    javah命令详解「建议收藏」概述:最近在写c++/c的一个小的项目,需要打成动态库,供java使用。就对java调用c++/c代码做了简答了解,在此做记录。jni开发第一步,就是用javah命令生成生成c\c++头文件。javah命令参数详解cmd(默认配置jdkpath)执行javah-help如下图:-d和-o这两个参数用于设置生成的C\C++头文件的指定,该两参数选项不能同时使…

    2022年9月25日
    3
  • php 递归算法

    php 递归算法通过递归实现阶乘functionmulti($n){

    2022年7月2日
    21
  • Flutter一步实现Toast 弹窗(fluttertoast 库)

    Flutter一步实现Toast 弹窗(fluttertoast 库)Flutter一步实现Toast弹窗(fluttertoast库)在Flutter中,想要实现一个Toast弹窗效果,最简单、最直接的方法就是使用fluttertoast库来实现。官网地址:https://pub.flutter-io.cn/packages/fluttertoast如何使用?导入依赖#addthislinetoyourdependenciesfluttertoast:^7.1.6在使用的地方,导入类import’package:f

    2022年9月24日
    6
  • 一阶惯性滤波特点_一阶惯性环节仿真

    一阶惯性滤波特点_一阶惯性环节仿真一阶惯性环节采用后置反馈的方式可以实现较精确的系统跟踪性能。上述系统的传递函数为因此启动性能良好,另,一阶惯性环节无超调量,因此可通过修改反馈参数实现最优的跟踪性能。因此在针对温度等变化较小的物理量方面的控制上是较占优势的,但精确跟踪也就意味着出现高频干扰、低频干扰、白噪声时,传感器也会精确地将这些干扰输出。这对一些容易受到干扰的系统是极为不利的。如下图为加入高频正弦信号后上述系统的输出(幅值为1,频率为1000(rad/sec))可见,系统虽然有一定的滤波性能(正弦输出

    2022年10月4日
    3
  • 二进制加,减法,23个位运算技巧[通俗易懂]

    二进制加,减法,23个位运算技巧[通俗易懂]二进制加,减法二进制最高位为1时表示负数,为0时表示正数。**原码:**一个正数,转换为二进制位就是这个正数的原码。负数的绝对值转换成二进制位然后在高位补1就是这个负数的原码。举例说明:      int类型的3的原码是11B(B表示二进制位),在32位机器上占四个字节,那么高位补零就得:      00000000000000000000000000000011    …

    2022年6月25日
    35

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号