tft每日頭條

 > 科技

 > android原生相機分析

android原生相機分析

科技 更新时间:2024-12-28 08:12:21
前言

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

何為前台程序員,是面向用戶的一端,包括前端、移動端、PC等等。

何為适配,适配就是當我們的開發環境、運行環境等發生變化的時候,程序依然能穩健運行。

而适配中最難為程序員的就是Android了,除了開發環境、運行環境等因素之外,因為Android開源的原因,還要适配各大廠商。。

而适配條件之多,經常讓Android程序員為之頭疼。

來看看相機、相冊相關的适配曆程:

  • Android 6 權限适配
  • Android 7 文件适配
  • Android 10/11 存儲适配

ok,接下來以一個更換頭像的小例子來講解一下。

示例

android原生相機分析(11适配指南之系統相機拍照)1

點擊頭像,然後彈窗,給出不同的選項,執行不同的操作。

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!!) } ... } } }

在requestCode是REQUEST_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 開發者手冊》

參考
  • 官方相機文檔
  • 官方權限文檔
  • 官方存儲文檔
,

更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!

查看全部

相关科技资讯推荐

热门科技资讯推荐

网友关注

Copyright 2023-2024 - www.tftnews.com All Rights Reserved