Android Camera2獲取預覽尺寸和fps範圍

升降攝像頭安卓手機剛上市的時候,有些很流行的app剛打開時,前置攝像頭就升起來了。好像就是出來看一眼然後又收回去。
雖然我們不調用拍照功能,只是爲了獲取相機的信息,也是可能讓攝像頭升起來的。

Camera實現

使用android.hardware.Camera獲取攝像頭支持的預覽尺寸和fps。
Camera.open獲取到camera實例,然後再camera.getParameters()

for (int i = 0; i < Camera.getNumberOfCameras(); i++) {
    Camera camera = null;
    try {
        camera = Camera.open(i);
        Parameters parameters = camera.getParameters();
        supportedSizes = parameters.getSupportedPreviewSizes();
        supportedFpsRanges = parameters.getSupportedPreviewFpsRange();
    } catch (RuntimeException e) {
        Log.e(TAG, "Failed to open, skipping", e);
        continue;
    } finally {
        if (camera != null) {
            camera.release();
        }
    }
}

android.hardware.Camera:下文簡稱爲「camera」或者「camera1」。

上面的代碼對於可升降攝像頭手機(例如榮耀X10)來說,前置攝像頭會升起來,然後又縮回去。用戶體驗不好。

榮耀X10前置攝像頭

可以用Camera2來實現,不用打開攝像頭就能獲取到相關信息。

Camera2

引入camera2

首先看模塊的gradle。本文示例是放在模塊裏的。
我們需要引入androidx相關的包。

  • compileSdk 31
  • androidx.camera:camera-core:1.1.0-beta01
  • androidx.camera:camera-camera2:1.1.0-beta01
apply plugin: 'com.android.library'

android {
    compileSdk 31
    defaultConfig {
        minSdkVersion 19
        targetSdkVersion 31
        versionCode 110
        versionName "1.1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }

    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    implementation fileTree(include: ['*.jar'], dir: 'libs')
    implementation 'androidx.appcompat:appcompat:1.3.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'com.android.support.test:runner:1.0.2'
    androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'

    implementation "androidx.camera:camera-core:1.1.0-beta01"
    implementation "androidx.camera:camera-camera2:1.1.0-beta01"
}

注意這裏minSdkVersion 19,後面使用的時候需要判斷一下運行的Android版本

這裏使用的是beta01版的camera2包,可以看出camera2仍在發展中

使用

部分導入的包的情況

import android.hardware.camera2.CameraAccessException;
import android.hardware.camera2.CameraCharacteristics;
import android.hardware.camera2.CameraManager;
import android.hardware.camera2.params.StreamConfigurationMap;
import android.os.Build;

用獲取系統服務的方式來獲取CameraManager,這裏要求 API >= 21

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
    CameraManager cameraManager = (CameraManager) mContext.getSystemService(Context.CAMERA_SERVICE);
}

Camera2相比於camera1的一大不同是,Camera2不需要調用Camera.open方法,就能去查詢相機的一些配置。

獲取支持的尺寸。android.hardware包裏的Camera.Size已經不推薦使用了(Deprecated)。Camera2用的是android.util.Size

List<android.util.Size> supportedUtilSize = new ArrayList<>();
CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics("0"); // 0和1
StreamConfigurationMap configs = characteristics.get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP);
supportedUtilSize = Arrays.asList(configs.getOutputSizes(ImageFormat.JPEG));

獲取支持的fps範圍,結果用Range來表示。

CameraCharacteristics characteristics = cameraManager.getCameraCharacteristics("1"); // 0和1
Range<Integer>[] ranges21 = characteristics.get(CameraCharacteristics.CONTROL_AE_AVAILABLE_TARGET_FPS_RANGES);

注意上面的代碼需要判斷系統版本

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
    // 在這裏面進行操作
}

Range介紹

獲取fps數據時我們用到了Range類。

package android.util;

public final class Range<T extends Comparable<? super T>> {

它接受的範型要能夠進行比較,即繼承Comparable。前面的代碼我們用的是Range

package java.lang;

public final class Integer extends Number implements Comparable<Integer> 

Range表示範圍。它代表了一個範圍,提供了一些常用的方法。比較類似數學中的集合的概念。

例如:

判斷目標值是否在範圍裏public boolean contains(T value)

判斷傳入的範圍是否在自己的範圍裏面public boolean contains(Range<T> range)

獲取一個範圍內的數值,即限制數值不能超過自己的範圍 public T clamp(T value)。假設範圍是0到10,傳入12,則返回10,也就是上限。

參考

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章