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 ?
You may have an issue with the USB camera driver and the camera selection rather than with permissions.
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.
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});
}