Android 使用 Camera2 以编程方式打开 on/off 相机闪光灯

Android Turn on/off Camera Flash Programmatically with Camera2

我正在制作一个用作手电筒的简单应用程序。我需要能够通过 Camera2 API 的应用程序中的按钮打开和关闭闪光灯,因为旧相机 API 出现错误。我想使用 "torch" 手电筒,无需打开相机即可打开闪光灯。我知道这是可能的,因为很多手机的快速设置菜单中都有闪光灯 on/off,但我找不到任何代码或教程来说明如何执行此操作。

代码:

Camera cam = Camera.open();     
Parameters p = cam.getParameters();
p.setFlashMode(Parameters.FLASH_MODE_TORCH);
cam.setParameters(p);
cam.startPreview();

我 运行 使用 Android 6.0.1 的 Nexus 6P。

编辑:我需要使用 Camera2 API。我还没有找到一个很好的教程,在得到答案后,我将在这里创建一个 Q/A 格式的教程。

我将向 Android 开发人员致辞。有关 CameraManager 的文档,因为需要更多代码,但这是在 API 21 以上的相机上激活闪光灯的基本代码。

CameraManager camManager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);
String cameraId = camManager.getCameraIdList()[0]; // Usually front camera is at 0 position.
camManager.setTorchMode(cameraId, true);

Android CameraManager documentation.

Android.hardware.camera2 documentation.

记住非常重要的一点,您需要使用 try/catch 来排除可能的错误,当然还要检查目前没有其他更高优先级的应用程序正在使用相机。

制作手电筒应用程序的最简单、最平庸的方法

// xml 代码

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.example.android.torcia.MainActivity">

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="ON"
        android:id="@+id/flash_button"
        android:layout_centerVertical="true"
        android:layout_centerHorizontal="true"
        android:onClick="turnOnOff"/>
</RelativeLayout>

//java代码

public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    Button flashButton;
    String buttonText;

    public void checkState(String buttonText) {


        switch (buttonText) {

            case "ON":

                flashButton.setText("OFF");
                flashOn();

                break;

            case "OFF":

                flashButton.setText("ON");
                flashOff();

                break;
        }
    }


    public void turnOnOff(View view) {

        flashButton = (Button) findViewById(R.id.flash_button);
        buttonText = flashButton.getText().toString();
        checkState(this.buttonText);

    }

    public void flashOn() {

        Camera camera = Camera.open();
        Camera.Parameters parameters = camera.getParameters();
        parameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
        camera.setParameters(parameters);
        camera.startPreview();

    }


    public void flashOff() {

        Camera camera2 = Camera.open();
        Camera.Parameters parameters2 = camera2.getParameters();
        parameters2.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
        camera2.setParameters(parameters2);
        camera2.stopPreview();

    }


}

// AndroidManifest

<?xml version="1.0" encoding="utf-8"?>

<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.FLASHLIGHT"/>

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/AppTheme">

    <activity android:name=".MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
</application>

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
    mCameraManager = (CameraManager) context.getSystemService(Context.CAMERA_SERVICE);
    try {
        for (String camID : mCameraManager.getCameraIdList()) {
            CameraCharacteristics cameraCharacteristics = mCameraManager.getCameraCharacteristics(camID);
            int lensFacing = cameraCharacteristics.get(CameraCharacteristics.LENS_FACING);
            if (lensFacing == CameraCharacteristics.LENS_FACING_FRONT && cameraCharacteristics.get(CameraCharacteristics.FLASH_INFO_AVAILABLE)) {
                mCameraId = camID;
                break;
            } else if (lensFacing == CameraCharacteristics.LENS_FACING_BACK && cameraCharacteristics.get(CameraCharacteristics.FLASH_INFO_AVAILABLE)) {
                mCameraId = camID;
            }
        }
        if (mCameraId != null) {
            mCameraManager.setTorchMode(mCameraId, true);
        }
    } catch (CameraAccessException e) {
        e.printStackTrace();
    }
}

mCameraId优先开启前置闪光灯,如果没有前置闪光灯则使用后置闪光灯。如果设备没有闪光灯,则 mCameraId 将为空,并且不会在上面的代码中调用 setTorchMode。

试试这个。 android Pie 及以上版本对我来说效果很好。有一个开关按钮可以打开和关闭手电筒

    ToggleButton FlB = (ToggleButton) findViewById(R.id.FlashBt);

    final boolean[] IsTurnedOn = {false};
    final CameraManager camManager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);
    String cameraId = null; // Usually front camera is at 0 position.
    try {
        cameraId = camManager.getCameraIdList()[0];
    } catch (CameraAccessException e) {
        e.printStackTrace();
    }
    final String finalCameraId = cameraId;

    FlB.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            if (!IsTurnedOn[0]){
                try {
                    camManager.setTorchMode(finalCameraId, true);
                } catch (CameraAccessException e) {
                    e.printStackTrace();
                }
            }
            else {
                try {
                    camManager.setTorchMode(finalCameraId, false);
                } catch (CameraAccessException e) {
                    e.printStackTrace();
                }
            }
            IsTurnedOn[0] =!IsTurnedOn[0];

        }
    });