29

我需要一种在录制视频时控制 Android 设备上的相机闪光灯的方法。我正在制作一个频闪灯应用程序,使用闪烁的频闪灯拍摄视频将能够记录高速移动的物体,例如风扇叶片。

只能通过启动视频预览并在相机参数中设置 FLASH_MODE_TORCH 来启用闪光灯。看起来像这样:

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

预览开始后,我可以来回翻转该参数以打开和关闭灯。在我尝试录制视频之前,这很有效。麻烦的是,为了把摄像头交给 MediaRecorder,我首先要解锁它。

MediaRecorder m = new MediaRecorder();
c.unlock();     // the killer
m.setCamera(c);

解锁后,我无法再更改相机参数,因此无法更改闪光灯状态。

我不知道是否真的可以做到这一点,因为我不是最擅长 java-hacking,但这是我所知道的:

  • Camera.unlock() 是本机方法,因此我无法真正看到它锁定我的方式背后的机制
  • Camera.Parameter 有一个包含其所有参数的 HashMap
  • Camera.setParameters(Parameters) 获取 HashMap,将其转换为字符串,并将其传递给本地方法
  • 我可以从 HashMap 中消除所有参数,但 TORCH-MODE 并且相机仍然会接受它

所以,我仍然可以访问相机,但它不会听我告诉它的任何内容。(这就是 Camera.unlock() 的目的)

编辑:

检查本机代码后,我可以看到在CameraService.cpp中我对 Camera.setParameters(Parameters) 的调用被拒绝,因为我的进程 ID 与相机服务记录的进程 ID 不匹配。所以这似乎是我的障碍。

编辑2:

看起来MediaPlayerService是在录制视频时控制相机的主要服务。我不知道这是否可能,但如果我能以某种方式在我自己的进程中启动该服务,我应该能够跳过 Camera.unlock() 调用。

编辑3:

最后一个选择是如果我能以某种方式获得指向 CameraHardwareInterface 的指针。从外观上看,这是一个特定于设备的接口,可能不包括 PID 检查。不过,主要问题是我唯一能找到指向它的指针的地方是在 CameraService 中,而 CameraService 没有说话。

Edit4:(几个月后)

在这一点上,我认为不可能做到我最初想要的。我不想在有人回答的情况下删除该问题,但我并没有积极寻求答案。(虽然,收到一个有效的答案会很棒。)

4

4 回答 4

15

我遇到了类似的问题。用户应该能够在录制过程中根据光线情况更改闪光模式以满足他们的需要。经过一些调查研究,我得出以下解决方案:

我假设您已经设置了适当的 SurfaceView 和 SurfaceHolder 及其必要的回调。我做的第一件事是提供这段代码(未声明的变量是全局变量):

public void surfaceCreated(SurfaceHolder holder) {
    try {
        camera = Camera.open();

        parameters = camera.getParameters();
        parameters.setFlashMode(Parameters.FLASH_MODE_OFF);

        camera.setParameters(parameters);
        camera.setPreviewDisplay(holder);
        camera.startPreview();

        recorder = new MediaRecorder();
    } catch (IOException e) {
        e.printStackTrace();
    }       
}

我的下一步是初始化和准备录音机:

private void initialize() {
    camera.unlock();

    recorder.setCamera(camera);
    recorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER);
    recorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
    recorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
    recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AAC);
    recorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);
    recorder.setVideoFrameRate(20);
    recorder.setOutputFile(filePath);

    try {
        recorder.prepare();
    } catch (IllegalStateException e) {
        e.printStackTrace();
        finish();
    } catch (IOException e) {
        e.printStackTrace();
        finish();
    }
}

需要注意的是,camera.unlock() 必须在媒体记录器的整个初始化过程之前调用。也就是说,还要注意每个 set 属性的正确顺序,否则在调用 prepare() 或 start() 时会出现 IllegalStateException。说到录音,我就是这么做的。这通常由视图元素触发:

public void record(View view) {
    if (recording) {
        recorder.stop();

        //TODO: do stuff....

        recording = false;
    } else {
        recording = true;

        initialize();
        recorder.start();
    }
}

所以现在,我终于可以正常录制了。但是那个闪光灯是怎么回事?最后但并非最不重要的是,这里有幕后的魔力:

public void flash(View view) {
    if(!recording) {
        camera.lock();
    }

    parameters.setFlashMode(parameters.getFlashMode().equals(Parameters.FLASH_MODE_TORCH) ? Parameters.FLASH_MODE_OFF : Parameters.FLASH_MODE_TORCH);
    camera.setParameters(parameters);

    if(!recording) {
        camera.unlock();
    }
}

每次我通过 onClick 操作调用该方法时,我都可以更改闪光灯模式,即使在录制期间也是如此。只需注意正确锁定相机即可。一旦在录制过程中媒体记录器获得锁定,您就不必再次锁定/解锁相机。它甚至不起作用。这是在带有 Android 版本 4.1.2 的三星 Galaxy S3 上测试的。希望这种方法有所帮助。

于 2013-02-13T14:22:21.310 回答
1

准备好媒体记录器后,使用camera.lock(),然后设置你想要设置到camera的任何参数。但是在开始录制之前你需要调用camera.unlock(),在你停止媒体录制之后你需要调用camera.lock()来开始预览。享受!!!

于 2015-03-05T11:18:24.097 回答
0

试试这个..希望它会工作.. :)

 private static Torch torch;

          public Torch() {
            super();
            torch = this;
          }

          public static Torch getTorch() {
            return torch;
          }

          private void getCamera() {
            if (mCamera == null) {
              try {
                mCamera = Camera.open();
              } catch (RuntimeException e) {
                Log.e(TAG, "Camera.open() failed: " + e.getMessage());
              }
            }
          }
        public void toggleLight(View view) {
            toggleLight();
          }

          private void toggleLight() {
            if (lightOn) {
              turnLightOff();
            } else {
              turnLightOn();
            }
          }

          private void turnLightOn() {
            if (!eulaAgreed) {
              return;
            }
            if (mCamera == null) {
              Toast.makeText(this, "Camera not found", Toast.LENGTH_LONG);
                   button.setBackgroundColor(COLOR_WHITE);
              return;
            }
            lightOn = true;
            Parameters parameters = mCamera.getParameters();
            if (parameters == null) {
                    button.setBackgroundColor(COLOR_WHITE);
              return;
         }
            List<String> flashModes = parameters.getSupportedFlashModes();
               if (flashModes == null) {
                   button.setBackgroundColor(COLOR_WHITE);
              return;
            }
            String flashMode = parameters.getFlashMode();
            Log.i(TAG, "Flash mode: " + flashMode);
            Log.i(TAG, "Flash modes: " + flashModes);
            if (!Parameters.FLASH_MODE_TORCH.equals(flashMode)) {
                   if (flashModes.contains(Parameters.FLASH_MODE_TORCH)) {
                parameters.setFlashMode(Parameters.FLASH_MODE_TORCH);
                mCamera.setParameters(parameters);
                button.setBackgroundColor(COLOR_LIGHT);
                startWakeLock();
              } else {
                Toast.makeText(this, "Flash mode (torch) not supported",
                    Toast.LENGTH_LONG);
                       button.setBackgroundColor(COLOR_WHITE);
                Log.e(TAG, "FLASH_MODE_TORCH not supported");
              }
            }
          }
         private void turnLightOff() {
            if (lightOn) {
                    button.setBackgroundColor(COLOR_DARK);
              lightOn = false;
              if (mCamera == null) {
                return;
              }
              Parameters parameters = mCamera.getParameters();
              if (parameters == null) {
                return;
              }
              List<String> flashModes = parameters.getSupportedFlashModes();
              String flashMode = parameters.getFlashMode();
                   if (flashModes == null) {
                return;
              }
              Log.i(TAG, "Flash mode: " + flashMode);
              Log.i(TAG, "Flash modes: " + flashModes);
              if (!Parameters.FLASH_MODE_OFF.equals(flashMode)) {
                       if (flashModes.contains(Parameters.FLASH_MODE_OFF)) {
                  parameters.setFlashMode(Parameters.FLASH_MODE_OFF);
                  mCamera.setParameters(parameters);
                  stopWakeLock();
                } else {
                  Log.e(TAG, "FLASH_MODE_OFF not supported");
                }
              }
            }
          }
     private void startPreview() {
        if (!previewOn && mCamera != null) {
          mCamera.startPreview();
          previewOn = true;
        }
      }

      private void stopPreview() {
        if (previewOn && mCamera != null) {
          mCamera.stopPreview();
          previewOn = false;
        }
      }

      private void startWakeLock() {
        if (wakeLock == null) {
          Log.d(TAG, "wakeLock is null, getting a new WakeLock");
          PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
          Log.d(TAG, "PowerManager acquired");
          wakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, WAKE_LOCK_TAG);
          Log.d(TAG, "WakeLock set");
        }
        wakeLock.acquire();
        Log.d(TAG, "WakeLock acquired");
      }

      private void stopWakeLock() {
        if (wakeLock != null) {
          wakeLock.release();
          Log.d(TAG, "WakeLock released");
        }
      }
     @Override
      public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (Eula.show(this)) {
          eulaAgreed = true;
        }
        setContentView(R.layout.main);
        button = findViewById(R.id.button);
        surfaceView = (SurfaceView) this.findViewById(R.id.surfaceview);
        surfaceHolder = surfaceView.getHolder();
        surfaceHolder.addCallback(this);
        surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
        disablePhoneSleep();
        Log.i(TAG, "onCreate");
      }
于 2012-11-28T06:49:59.533 回答
-3

要访问设备摄像头,您必须在 Android Manifest 中声明 CAMERA 权限。还要确保包含<uses-feature>manifest 元素以声明应用程序使用的相机功能。例如,如果您使用相机和自动对焦功能,您的清单应包括以下内容:

 <uses-permission android:name="android.permission.CAMERA" />
 <uses-feature android:name="android.hardware.camera" />
 <uses-feature android:name="android.hardware.camera.autofocus" />

检查手电筒支持的示例可能如下所示:

//Create camera and parameter objects
private Camera mCamera;
private Camera.Parameters mParameters;
private boolean mbTorchEnabled = false;

//... later in a click handler or other location, assuming that the mCamera object has already been instantiated with Camera.open()
mParameters = mCamera.getParameters();

//Get supported flash modes
List flashModes = mParameters.getSupportedFlashModes ();

//Make sure that torch mode is supported
//EDIT - wrong and dangerous to check for torch support this way
//if(flashModes != null && flashModes.contains("torch")){
if(flashModes != null && flashModes.contains(Camera.Parameters.FLASH_MODE_TORCH)){
    if(mbTorchEnabled){
        //Set the flash parameter to off
        mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
    }
    else{
        //Set the flash parameter to use the torch
        mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
    }

    //Commit the camera parameters
    mCamera.setParameters(mParameters);

    mbTorchEnabled = !mbTorchEnabled;
}

要打开手电筒,您只需设置相机参数Camera.Parameters.FLASH_MODE_TORCH

Camera mCamera;
Camera.Parameters mParameters;

//Get a reference to the camera/parameters
mCamera = Camera.open();
mParameters = mCamera.getParameters();

//Set the torch parameter
mParameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);

//Comit camera parameters
mCamera.setParameters(mParameters);

要关闭手电筒,请设置Camera.Parameters.FLASH_MODE_OFF

于 2012-09-03T17:49:00.283 回答