Search code examples
androidwebviewusb-camera

Access an external USB camera from the webview in Android


I have a webview in my Android app. From the webview, I am able to access default camera on the phone. The following code works fine.

webview.setWebChromeClient(new WebChromeClient(){
    // Need to accept permissions to use the camera
    @Override
    public void onPermissionRequest(final PermissionRequest request) {
        L.d("onPermissionRequest");
        request.grant(request.getResources());
    }
});

My issue is to access an external USB camera from this webview. Above permission opens up the default camera in the webview.

I haven't been able to find any documentation which even says this thing is possible. Is it because of some security issue ?


Solution

  • You may have an issue with the USB camera driver and the camera selection rather than with permissions.

    1. USB Camera Drivers

    The first step is to make sure that your USB camera is being detected and is working on your device. You didn't indicate whether you had confirmed this or not. My understanding is that android.hardware.camera2 support for USB cameras is still rather poor. If your camera is supported then hopefully it will enumerate along with the rest of the cameras. In my test with Android 8.1.0 the USB camera I attached was not enumerated with CameraManager, while it was with the library below.

    The USB Camera library https://github.com/saki4510t/UVCCamera is frequently used to provide broader support for USB cameras, but from my limited experience with the library it writes to a TextureView and so will probably not play nicely with WebRTC in a WebView. In a very cursory investigation I did not see WebView hooks that would support connecting an external video source.

    1. Camera Selection

    You mentioned that your test always used the default camera so it sounds like you might not be actively enumerating and selecting the target camera. WebRTC camera selection can be performed in Javascript using the navigator.mediaDevices interface. For example,

    function chooseDevice(videoInDevices) {
        // return selected device
    }
    
    // Filter devices so we only consider video sources
    function filterForVideoInputs(devices) {
        return devices.filter(d => d.kind === 'videoinput');
    }
    
    // Simply pull out deviceId from selected device struct
    function getDeviceId(deviceInfo) {
        return deviceInfo.deviceId;
    }
    
    // Request video stream from selected deviceId
    function requestDevice(deviceId) {
        return navigator.mediaDevices.getUserMedia({
            video: {
                deviceId: {
                    exact: deviceId
                }
            }
        });
    }
    
    // Connect stream from getUserMedia to HTML5 video element
    function startStream(stream) {
        let video = document.querySelector('video');
        video.srcObject = stream;
        video.onloadedmetadata = function () {
            video.play();
        }
    }
    
    navigator.mediaDevices.enumerateDevices()
        .then(filterForVideoInputs)
        .then(chooseDevice)
        .then(getDeviceId)
        .then(requestDevice)
        .then(startStream)
        .catch(err => console.log(err));
    

    Since your permissions were sufficient for the internal camera, they should as far as I understand also be sufficient for a USB camera.

    Note, as you may know, the documentation has a warning about not blinding granting Webkit permission. When you move toward production remember to replace

    request.grant(request.getResources())
    

    perhaps with something more like this

    if (isRequestOriginOrWhateverApproved(request)) {
        request.grant(new String[]{PermissionRequest.RESOURCE_VIDEO_CAPTURE});
    }