是否可以通过网站控制手机上的相机灯?

IT技术 javascript google-chrome mobile getusermedia
2021-03-04 04:35:48

是否可以通过网站控制手机上的相机灯?通过 Chrome 或 Firefox 说。我知道可以使用 Android 或 iOS 应用程序,那里有许多手电筒应用程序。而且我知道可以通过 getUserMedia 函数系列控制相机。如果没有,有谁知道它什么时候可用?

3个回答

这是一个网站的小“火炬应用程序”:

编辑 1:我也做了一个jsfiddle

//Test browser support
const SUPPORTS_MEDIA_DEVICES = 'mediaDevices' in navigator;

if (SUPPORTS_MEDIA_DEVICES) {
  //Get the environment camera (usually the second one)
  navigator.mediaDevices.enumerateDevices().then(devices => {
  
    const cameras = devices.filter((device) => device.kind === 'videoinput');

    if (cameras.length === 0) {
      throw 'No camera found on this device.';
    }
    const camera = cameras[cameras.length - 1];

    // Create stream and get video track
    navigator.mediaDevices.getUserMedia({
      video: {
        deviceId: camera.deviceId,
        facingMode: ['user', 'environment'],
        height: {ideal: 1080},
        width: {ideal: 1920}
      }
    }).then(stream => {
      const track = stream.getVideoTracks()[0];

      //Create image capture object and get camera capabilities
      const imageCapture = new ImageCapture(track)
      const photoCapabilities = imageCapture.getPhotoCapabilities().then(() => {

        //todo: check if camera has a torch

        //let there be light!
        const btn = document.querySelector('.switch');
        btn.addEventListener('click', function(){
          track.applyConstraints({
            advanced: [{torch: true}]
          });
        });
      });
    });
  });
  
  //The light will be on as long the track exists
  
  
}
<button class="switch">On / Off</button>

代码深受这个存储库、这个网络系列和这个博客文章的启发

编辑 2: 这仅适用于 Chrome(可能还有 Opera)。它在 iOS 上的Chrome 中不起作用因为 Chrome 无法访问相机我现在无法在 android 上测试它。我创建了一个带有输出的新 jsfiddle。如果您有一部 android 手机并且它不适合您,它可能会说明原因:https : //jsfiddle.net/jpa1vwed/

随意调试,评论和编辑。

这在iOS上有效吗?我有一个可用的应用程序在 iOS 上的 Safari 11 中使用 getUserMedia 访问相机,但剪辑对我不起作用。有没有人测试过这个?
2021-04-22 04:35:48
Chrome 无法访问 iphone 上的相机(stackoverflow.com/questions/51501642/...),所以这也不起作用......我将在 android 上的 chrome 中测试它,我需要先找到一个 android 手机。
2021-04-27 04:35:48
有人可以更新它以向火炬添加切换功能吗?就像一旦打开手电筒一样,就无法将其关闭
2021-05-07 04:35:48
@DanielBudick Chrome 可以并且确实访问 iPhone 上的两个摄像头,您需要使用正确的 html5 输入标签<input type="file" accept="image/*" capture="environment">“环境”用于背面,“用户”用于正面。
2021-05-08 04:35:48
它只能在 Chrome 中工作,因为ImageCapture目前仅在 Chrome 和 Opera 中受支持(我没有安装 Opera,因此无法对其进行测试)。它不再适用于 Iphone 上的 chrome,我目前正在研究它。
2021-05-14 04:35:48

您可以使用MediaStream图像捕捉API通过创建ImageCapture中VideoStreamTrack和设置选项“fillLightMode”“闪”“火炬”例子:

<video autoplay="true"></video>
<img />
<button onclick="takePhoto()">Take Photo</button>
<script type="text/javascript">
    var imageCapture = null;
    var deviceConfig = {
        video: {
            width: 480,
            height: 640,
            facingMode: "environment", /* may not work, see https://bugs.chromium.org/p/chromium/issues/detail?id=290161 */
            deviceId: null
        }
    };

    var imageCaptureConfig = {
        fillLightMode: "torch", /* or "flash" */
        focusMode: "continuous"
    };

    // get the available video input devices and choose the one that represents the backside camera
    navigator.mediaDevices.enumerateDevices()
            /* replacement for not working "facingMode: 'environment'": use filter to get the backside camera with the flash light */
            .then(mediaDeviceInfos => mediaDeviceInfos.filter(mediaDeviceInfo => ((mediaDeviceInfo.kind === 'videoinput')/* && mediaDeviceInfo.label.includes("back")*/)))
            .then(mediaDeviceInfos => {
                console.log("mediaDeviceInfos[0].label: " + mediaDeviceInfos[0].label);

                // get the device ID of the backside camera and use it for media stream initialization
                deviceConfig.video.deviceId = mediaDeviceInfos[0].deviceId;
                navigator.mediaDevices.getUserMedia(deviceConfig)
                        .then(_gotMedia)
                        .catch(err => console.error('getUserMedia() failed: ', err));
            });

    function takePhoto () {
        imageCapture.takePhoto()
                .then(blob => {
                    console.log('Photo taken: ' + blob.type + ', ' + blob.size + 'B');

                    // get URL for blob data and use as source for the image element
                    const image = document.querySelector('img');
                    image.src = URL.createObjectURL(blob);
                })
                .catch(err => console.error('takePhoto() failed: ', err));
    }

    function _gotMedia (mediastream) {
        // use the media stream as source for the video element
        const video = document.querySelector('video');
        video.srcObject = mediastream;

        // create an ImageCapture from the first video track
        const track = mediastream.getVideoTracks()[0];
        imageCapture = new ImageCapture(track);

        // set the image capture options (e.g. flash light, autofocus, ...)
        imageCapture.setOptions(imageCaptureConfig)
                .catch(err => console.error('setOptions(' + JSON.stringify(imageCaptureConfig) + ') failed: ', err));
    }
</script>

笔记:

  • 在撰写本文时,API 仍在开发中,将来可能会发生变化。
  • 在 Chrome 中启用ImageCapture,必须将标志“chrome://flags/#enable-experimental-web-platform-features”设置为“true”
  • 为使ImageCapture中的Firefox中的标志“dom.imagecapture.enabled”“about:config中”已被设置为“真”但是在撰写本文时不支持“setOptions”

也可以看看:

四年过去了,已经是2021年了,这个功能好像没有了。可以理解。ImageCapture支持比底层轨道更多的操作,例如无需再次请求许可即可关闭和打开手电筒。也许根本不可能支持所有设备,所以他们不得不修剪它。
2021-04-28 04:35:48

我修复了 Daniel 的答案,现在按钮可以在 Android 手机上正常工作。 IOS 仍然不受支持。

https://jsfiddle.net/nzw5tv1q/