Android 之 CameraX 初體驗(yàn)

CameraX 是一個(gè) Jetpack 支持庫(kù)蒸甜,利用了 camera2 的功能棠耕,可感知生命周期,解決了設(shè)備兼容性問題柠新,向后兼容至 Android 5.0(API 級(jí)別 21)窍荧,并提供一致且易用的 API 接口。借助 CameraX登颓,開發(fā)者只需兩行代碼就能實(shí)現(xiàn)與預(yù)安裝的相機(jī)應(yīng)用相同的相機(jī)體驗(yàn)和功能搅荞!

它是一種架構(gòu),官方介紹它是非常強(qiáng)大好用,那么咕痛,嘗試用起來(lái)吧痢甘。

  • 1、創(chuàng)建 CameraX 的練習(xí)項(xiàng)目

minSdkVersion 選 21茉贡,Android 5.0塞栅。

  • 2、添加依賴
    // CameraX core library using camera2 implementation
    implementation "androidx.camera:camera-camera2:1.0.1"
    // CameraX Lifecycle Library
    implementation "androidx.camera:camera-lifecycle:1.0.1"
    // CameraX View class
    implementation "androidx.camera:camera-view:1.0.0-alpha27"
  • 3腔丧、添加插件 kotlin-android-extensions
plugins {
    id 'com.android.application'
    id 'kotlin-android'
    id 'kotlin-android-extensions'
}
  • 4放椰、編寫布局文件 activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <Button
        android:id="@+id/camera_capture_button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginBottom="30dp"
        android:elevation="2dp"
        android:scaleType="fitCenter"
        android:text="Take Photo"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent" />

    <androidx.camera.view.PreviewView
        android:id="@+id/viewFinder"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</androidx.constraintlayout.widget.ConstraintLayout>
  • 5、打開 AndroidManifest.xml 添加權(quán)限
    <!-- 確保設(shè)備有 camera .any 意味著可能是前置攝像頭或者是后置攝像頭  -->
    <uses-feature android:name="android.hardware.camera.any" />
    <uses-permission android:name="android.permission.CAMERA" />

6愉粤、正式編寫代碼啦砾医,包括了打開相機(jī),截取圖像衣厘,保存圖片如蚜,圖片分析等等功能編寫。

步驟:

  • 將 CameraX 依賴項(xiàng)加入到的項(xiàng)目中影暴。
  • 顯示相機(jī)取景器(使用預(yù)覽用例)错邦。
  • 實(shí)現(xiàn)照片捕獲,將圖像保存到存儲(chǔ)(使用 ImageCapture 用例)型宙。
  • 實(shí)時(shí)分析來(lái)自相機(jī)的幀(使用 ImageAnalysis 用例)撬呢。
package com.pyn.cameraxpractice

import android.Manifest
import android.content.pm.PackageManager
import android.net.Uri
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.util.Log
import android.widget.Toast
import androidx.camera.core.*
import androidx.camera.lifecycle.ProcessCameraProvider
import androidx.core.app.ActivityCompat
import androidx.core.content.ContextCompat
import com.pyn.cameraxpractice.databinding.ActivityMainBinding
import kotlinx.android.synthetic.main.activity_main.*
import java.io.File
import java.nio.ByteBuffer
import java.text.SimpleDateFormat
import java.util.*
import java.util.concurrent.ExecutorService
import java.util.concurrent.Executors

class MainActivity : AppCompatActivity() {

    private lateinit var mBinding: ActivityMainBinding
    private var imageCapture: ImageCapture? = null
    private lateinit var outputDirectory: File
    private lateinit var cameraExecutor: ExecutorService

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        mBinding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(mBinding.root)

        if (allPermissionsGranted()) {
            // 權(quán)限請(qǐng)求完畢,且授權(quán)了
            startCamera()
        } else {
            ActivityCompat.requestPermissions(this, REQUIRED_PERMISSIONS, REQUEST_CODE_PERMISSIONS)
        }

        mBinding.btnCameraCapture.setOnClickListener { takePhoto() }
        outputDirectory = getOutputDirectory()
        cameraExecutor = Executors.newSingleThreadExecutor()
    }

    /**
     * 請(qǐng)求權(quán)限的回調(diào)
     */
    override fun onRequestPermissionsResult(
        requestCode: Int, permissions: Array<String>, grantResults:
        IntArray
    ) {
        if (requestCode == REQUEST_CODE_PERMISSIONS) {
            if (allPermissionsGranted()) {
                // 如果所有權(quán)限都授權(quán)成功了
                startCamera()
            } else {
                Toast.makeText(
                    this,
                    "Permissions not granted by the user.",
                    Toast.LENGTH_SHORT
                ).show()
                finish()
            }
        }
    }

    /**
     * 拍照
     */
    private fun takePhoto() {
        // 獲取對(duì) ImageCapture 用例的引用
        val imageCapture = imageCapture ?: return
        // 創(chuàng)建文件來(lái)保存圖像妆兑。添加時(shí)間戳跨算,以便文件名是唯一的遥椿。
        val photoFile = File(
            outputDirectory,
            SimpleDateFormat(FILENAME_FORMAT, Locale.US).format(System.currentTimeMillis()) + ".jpg"
        )
        // 指定將輸出內(nèi)容保存在我們剛剛創(chuàng)建的文件中
        val outputOptions = ImageCapture.OutputFileOptions.Builder(photoFile).build()
        // 拍照
        imageCapture.takePicture(
            outputOptions,
            ContextCompat.getMainExecutor(this),
            object : ImageCapture.OnImageSavedCallback {
                override fun onError(exception: ImageCaptureException) {
                    Log.e(TAG, "Photo capture failed: ${exception.message}", exception)
                }

                override fun onImageSaved(outputFileResults: ImageCapture.OutputFileResults) {
                    val savedUri = Uri.fromFile(photoFile)
                    val msg = "Photo capture succeeded: $savedUri"
                    Toast.makeText(baseContext, msg, Toast.LENGTH_SHORT).show()
                    Log.d(TAG, msg)
                }
            }
        )
    }

    /**
     * 打開相機(jī)
     */
    private fun startCamera() {

        // 定義配置链嘀,創(chuàng)建用例實(shí)例兜叨,用于將相機(jī)的生命周期綁定到生命周期所有者,這消除了打開和關(guān)閉相機(jī)的任務(wù)谱姓,因?yàn)?CameraX 具有生命周期感知能力借尿。
        val cameraProviderFuture = ProcessCameraProvider.getInstance(this)
        // 向 cameraProviderFuture 添加一個(gè)監(jiān)聽器。
        // 參數(shù)一:Runnable
        // 參數(shù)二:ContextCompat.getMainExecutor() 將返回一個(gè)在主線程上運(yùn)行的 Executor屉来。
        cameraProviderFuture.addListener(Runnable {
            // 用于將相機(jī)的生命周期綁定到應(yīng)用程序進(jìn)程中的 LifecycleOwner
            val cameraProvider: ProcessCameraProvider = cameraProviderFuture.get()
            // 初始化預(yù)覽對(duì)象路翻,取景器是用來(lái)讓用戶預(yù)覽照片的,CameraX Preview 實(shí)現(xiàn)取景器
            val preview = Preview.Builder().build().also {
                it.setSurfaceProvider(viewFinder.surfaceProvider)
            }
            imageCapture = ImageCapture.Builder().build()

            val imageAnalyzer = ImageAnalysis.Builder()
                .build()
                .also {
                    it.setAnalyzer(cameraExecutor, LuminosityAnalyzer { luma ->
                        Log.d(TAG, "Average luminosity: $luma")
                    })
                }

            // 默認(rèn)選擇后置攝像頭
            val cameraSelector = CameraSelector.DEFAULT_BACK_CAMERA

            try {
                // 在重新綁定之前取消綁定用例
                cameraProvider.unbindAll()
                // 將cameraSelector 和預(yù)覽對(duì)象綁定到 cameraProvider
                cameraProvider.bindToLifecycle(this, cameraSelector, preview, imageCapture, imageAnalyzer)
            } catch (exc: Exception) {
                // 綁定失敗
                Log.e(TAG, "Use case binding failed", exc)
            }
        }, ContextCompat.getMainExecutor(this))

    }

    private fun allPermissionsGranted() = REQUIRED_PERMISSIONS.all {
        ContextCompat.checkSelfPermission(baseContext, it) == PackageManager.PERMISSION_GRANTED
    }

    private fun getOutputDirectory(): File {
        val mediaDir = externalMediaDirs.firstOrNull()?.let {
            File(it, resources.getString(R.string.app_name)).apply {
                mkdirs()
            }
        }
        return if (mediaDir != null && mediaDir.exists()) mediaDir else filesDir
    }

    override fun onDestroy() {
        super.onDestroy()
        cameraExecutor.shutdown()
    }

    companion object {
        private const val TAG = "CameraXBasic"
        private const val FILENAME_FORMAT = "yyyy-MM-dd-HH"
        private const val REQUEST_CODE_PERMISSIONS = 10
        private val REQUIRED_PERMISSIONS = arrayOf(Manifest.permission.CAMERA)
    }

    private class LuminosityAnalyzer(private val listener: LumaListener) : ImageAnalysis.Analyzer {

        private fun ByteBuffer.toByteArray(): ByteArray {
            // 倒帶
            rewind()
            val data = ByteArray(remaining())
            get(data)
            return data
        }

        override fun analyze(image: ImageProxy) {
            val buffer = image.planes[0].buffer
            val data = buffer.toByteArray()
            val pixels = data.map { it.toInt() and 0xFF }
            val luma = pixels.average()
            listener(luma)
            image.close()
        }
    }
}

typealias LumaListener = (luma: Double) -> Unit

保存了圖片后茄靠,可以從手機(jī)文件中找到該圖片茂契,地址如下圖:

照片位置

分析日志如下:

分析日志

上述只是簡(jiǎn)單照著官方文檔練習(xí),具體更為詳細(xì)的還是需要再多去看看官方文檔研究下的慨绳,api 很靈活掉冶。多嘗試嘗試真竖。


附:

CameraX 開發(fā)者社區(qū) 「論壇」:
https://groups.google.com/a/android.com/g/camerax-developers
CameraX API 最佳實(shí)踐:
https://github.com/android/camera-samples
官方介紹地址:
https://developer.android.com/training/camerax?hl=zh-cn

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市厌小,隨后出現(xiàn)的幾起案子恢共,更是在濱河造成了極大的恐慌,老刑警劉巖璧亚,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件讨韭,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡癣蟋,警方通過查閱死者的電腦和手機(jī)透硝,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)疯搅,“玉大人濒生,你說我怎么就攤上這事”玻” “怎么了甜攀?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)琐馆。 經(jīng)常有香客問我,道長(zhǎng)恒序,這世上最難降的妖魔是什么瘦麸? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮歧胁,結(jié)果婚禮上滋饲,老公的妹妹穿的比我還像新娘。我一直安慰自己喊巍,他們只是感情好屠缭,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著崭参,像睡著了一般呵曹。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上何暮,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天奄喂,我揣著相機(jī)與錄音,去河邊找鬼海洼。 笑死跨新,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的坏逢。 我是一名探鬼主播域帐,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼赘被,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了肖揣?” 一聲冷哼從身側(cè)響起帘腹,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎许饿,沒想到半個(gè)月后阳欲,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡陋率,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年球化,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片瓦糟。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡筒愚,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出菩浙,到底是詐尸還是另有隱情巢掺,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布劲蜻,位于F島的核電站陆淀,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏先嬉。R本人自食惡果不足惜轧苫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望疫蔓。 院中可真熱鬧含懊,春花似錦、人聲如沸衅胀。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)滚躯。三九已至雏门,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間哀九,已是汗流浹背剿配。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留阅束,地道東北人呼胚。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像息裸,于是被迫代替她去往敵國(guó)和親蝇更。 傳聞我的和親對(duì)象是個(gè)殘疾皇子沪编,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容