使用Camera2在Android上以编程方式打开/关闭相机闪光灯

4
我是一个简单的应用程序,作为手电筒。 我需要能够使用Camera2 API从应用程序中的按钮打开和关闭闪光灯,因为我在旧的相机API中遇到错误。 我想使用“torch”手电筒,其中相机不必打开即可打开闪光灯。 我知道这是可能的,因为许多手机在快速设置菜单中有闪光灯开/关,但我找不到任何关于如何做到这一点的代码或教程。
代码:
Camera cam = Camera.open();     
Parameters p = cam.getParameters();
p.setFlashMode(Parameters.FLASH_MODE_TORCH);
cam.setParameters(p);
cam.startPreview();

我使用装有安卓6.0.1的Nexus 6P手机。

编辑:我需要使用Camera2 API。目前我还没有找到好的教程,在得到答案后,我会在这里创建一个问答格式的教程。


1
看看这个链接可能会对你有所帮助。https://dev59.com/TG025IYBdhLWcg3wVkef - Kristo
是的。相机和Android.hardware的两个权限都需要。 - Benjamin Owen
帖子已经用代码编辑过了。 - Benjamin Owen
你知道在API 21以上的版本中,Camera类将无法使用吗?除此之外,让我先了解一下新的CameraManager,稍后会回复你的答案。 - Daniel Netzer
我知道相机API在较新版本的Android上不起作用。我只是没有找到一个好的Camera2 API教程。我会查看Android开发者文档,看看能否从中获取任何信息。 - Benjamin Owen
显示剩余5条评论
4个回答

22

我将引导您查阅有关CameraManager的Android Dev文档,因为需要更多代码,但这是在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文档

Android.hardware.camera2文档

非常重要的一点是要记得使用try/catch来处理可能出现的错误,并且当然要检查当前没有其他更高优先级的应用程序正在使用相机。


后置摄像头(带闪光灯的那个)是否在位置1? - Benjamin Owen
抱歉,我之前说的是后置摄像头,但你可以随时测试它,如果你阅读文档,还可以获取相机属性并查看该特定相机是否附带闪光灯。编程的小提示是,始终阅读文档并进行实践操作。 - Daniel Netzer
好的。我会尝试获取完整列表并将其编辑到我的帖子中。我现在正在测试您的代码... - Benjamin Owen
如果您满意,请将其标记为答案。:D - Daniel Netzer
所以我在我的旧设备上收到了找不到类的错误...Could not find class 'android.hardware.camera2.CameraManager'。有没有办法让它忽略这个?(如果SDK版本太低,我会跳过那段代码。) - Benjamin Owen
显示剩余5条评论

2
 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 将为 null,并且上述代码中不会调用 setTorchMode。


2
尝试这个。对于安卓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];

        }
    });

非常好!我将这个解决方案调整为一个带有选项勾选值的选项菜单,并调用一个函数。 - undefined

-1

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

// 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();

    }


}

// Android清单

<?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>


2
这不是使用API的正确方式,尽管这在许多设备上可以工作。在调用startPreview之前,API要求您设置预览目标,并且在许多设备(例如所有新型Nexus手机)上强制执行此操作。 - Eddy Talvala
4
答案不正确!问题涉及更新的Camera2 API,但答案使用较旧的相机API版本。 - Ayaz Alifov

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接