Commit df30065a authored by jamescook's avatar jamescook Committed by Commit bot

app_shell: Allow getUserMedia to select a capture device by id

Return the device if it exists. Still return the first available device
if getUserMedia() doesn't ask for one in particular.

BUG=407236
TEST=manual

Review URL: https://codereview.chromium.org/514463002

Cr-Commit-Position: refs/heads/master@{#292061}
parent 986f5917
...@@ -4,32 +4,43 @@ ...@@ -4,32 +4,43 @@
#include "extensions/shell/browser/media_capture_util.h" #include "extensions/shell/browser/media_capture_util.h"
#include <string>
#include "base/callback.h" #include "base/callback.h"
#include "base/logging.h" #include "base/logging.h"
#include "content/public/browser/media_capture_devices.h" #include "content/public/browser/media_capture_devices.h"
#include "extensions/common/permissions/permissions_data.h" #include "extensions/common/permissions/permissions_data.h"
using content::MediaCaptureDevices; using content::MediaCaptureDevices;
using content::MediaStreamDevice;
using content::MediaStreamDevices; using content::MediaStreamDevices;
using content::MediaStreamUI; using content::MediaStreamUI;
namespace extensions { namespace extensions {
const MediaStreamDevice* GetRequestedDeviceOrDefault(
const MediaStreamDevices& devices,
const std::string& requested_device_id) {
if (!requested_device_id.empty())
return devices.FindById(requested_device_id);
if (!devices.empty())
return &devices[0];
return NULL;
}
namespace media_capture_util { namespace media_capture_util {
// See also Chrome's MediaCaptureDevicesDispatcher. // See also Chrome's MediaCaptureDevicesDispatcher.
void GrantMediaStreamRequestWithFirstDevice( void GrantMediaStreamRequest(content::WebContents* web_contents,
content::WebContents* web_contents, const content::MediaStreamRequest& request,
const content::MediaStreamRequest& request, const content::MediaResponseCallback& callback,
const content::MediaResponseCallback& callback, const Extension* extension) {
const Extension* extension) {
// app_shell only supports audio and video capture, not tab or screen capture. // app_shell only supports audio and video capture, not tab or screen capture.
DCHECK(request.audio_type == content::MEDIA_DEVICE_AUDIO_CAPTURE || DCHECK(request.audio_type == content::MEDIA_DEVICE_AUDIO_CAPTURE ||
request.video_type == content::MEDIA_DEVICE_VIDEO_CAPTURE); request.video_type == content::MEDIA_DEVICE_VIDEO_CAPTURE);
// app_shell does not support requesting a specific device ID.
DCHECK(request.requested_audio_device_id.empty() &&
request.requested_video_device_id.empty());
MediaStreamDevices devices; MediaStreamDevices devices;
const PermissionsData* permissions_data = extension->permissions_data(); const PermissionsData* permissions_data = extension->permissions_data();
...@@ -39,11 +50,11 @@ void GrantMediaStreamRequestWithFirstDevice( ...@@ -39,11 +50,11 @@ void GrantMediaStreamRequestWithFirstDevice(
CHECK(permissions_data->HasAPIPermission(APIPermission::kAudioCapture)) CHECK(permissions_data->HasAPIPermission(APIPermission::kAudioCapture))
<< "Audio capture request but no audioCapture permission in manifest."; << "Audio capture request but no audioCapture permission in manifest.";
// Use first available audio capture device. const MediaStreamDevice* device = GetRequestedDeviceOrDefault(
const MediaStreamDevices& audio_devices = MediaCaptureDevices::GetInstance()->GetAudioCaptureDevices(),
MediaCaptureDevices::GetInstance()->GetAudioCaptureDevices(); request.requested_audio_device_id);
if (!audio_devices.empty()) if (device)
devices.push_back(audio_devices[0]); devices.push_back(*device);
} }
if (request.video_type == content::MEDIA_DEVICE_VIDEO_CAPTURE) { if (request.video_type == content::MEDIA_DEVICE_VIDEO_CAPTURE) {
...@@ -51,11 +62,11 @@ void GrantMediaStreamRequestWithFirstDevice( ...@@ -51,11 +62,11 @@ void GrantMediaStreamRequestWithFirstDevice(
CHECK(permissions_data->HasAPIPermission(APIPermission::kVideoCapture)) CHECK(permissions_data->HasAPIPermission(APIPermission::kVideoCapture))
<< "Video capture request but no videoCapture permission in manifest."; << "Video capture request but no videoCapture permission in manifest.";
// Use first available video capture device. const MediaStreamDevice* device = GetRequestedDeviceOrDefault(
const MediaStreamDevices& video_devices = MediaCaptureDevices::GetInstance()->GetVideoCaptureDevices(),
MediaCaptureDevices::GetInstance()->GetVideoCaptureDevices(); request.requested_video_device_id);
if (!video_devices.empty()) if (device)
devices.push_back(video_devices[0]); devices.push_back(*device);
} }
// TODO(jamescook): Should we show a recording icon somewhere? If so, where? // TODO(jamescook): Should we show a recording icon somewhere? If so, where?
......
...@@ -18,14 +18,15 @@ class Extension; ...@@ -18,14 +18,15 @@ class Extension;
namespace media_capture_util { namespace media_capture_util {
// Grants access to the first available audio capture device (if requested) and // Grants access to audio and video capture devices.
// the first available video capture device (if requested). Usually used as a // * If the caller requests specific device ids, grants access to those.
// helper for media capture ProcessMediaAccessRequest(). // * If the caller does not request specific ids, grants access to the first
void GrantMediaStreamRequestWithFirstDevice( // available device.
content::WebContents* web_contents, // Usually used as a helper for media capture ProcessMediaAccessRequest().
const content::MediaStreamRequest& request, void GrantMediaStreamRequest(content::WebContents* web_contents,
const content::MediaResponseCallback& callback, const content::MediaStreamRequest& request,
const Extension* extension); const content::MediaResponseCallback& callback,
const Extension* extension);
} // namespace media_capture_util } // namespace media_capture_util
} // namespace extensions } // namespace extensions
......
...@@ -65,8 +65,8 @@ void ShellAppWindow::RequestMediaAccessPermission( ...@@ -65,8 +65,8 @@ void ShellAppWindow::RequestMediaAccessPermission(
content::WebContents* web_contents, content::WebContents* web_contents,
const content::MediaStreamRequest& request, const content::MediaStreamRequest& request,
const content::MediaResponseCallback& callback) { const content::MediaResponseCallback& callback) {
// Allow access to the first microphone and/or camera. // Allow access to the microphone and/or camera.
media_capture_util::GrantMediaStreamRequestWithFirstDevice( media_capture_util::GrantMediaStreamRequest(
web_contents, request, callback, extension_); web_contents, request, callback, extension_);
} }
......
...@@ -47,8 +47,8 @@ void ShellExtensionHostDelegate::ProcessMediaAccessRequest( ...@@ -47,8 +47,8 @@ void ShellExtensionHostDelegate::ProcessMediaAccessRequest(
const content::MediaStreamRequest& request, const content::MediaStreamRequest& request,
const content::MediaResponseCallback& callback, const content::MediaResponseCallback& callback,
const Extension* extension) { const Extension* extension) {
// Allow access to the first microphone and/or camera. // Allow access to the microphone and/or camera.
media_capture_util::GrantMediaStreamRequestWithFirstDevice( media_capture_util::GrantMediaStreamRequest(
web_contents, request, callback, extension); web_contents, request, callback, extension);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment