1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/browser/media/media_stream_capture_indicator.h"
9 #include "base/logging.h"
10 #include "base/memory/scoped_ptr.h"
11 #include "base/prefs/pref_service.h"
12 #include "base/strings/utf_string_conversions.h"
13 #include "chrome/app/chrome_command_ids.h"
14 #include "chrome/browser/browser_process.h"
15 #include "chrome/browser/profiles/profile.h"
16 #include "chrome/browser/status_icons/status_icon.h"
17 #include "chrome/browser/status_icons/status_tray.h"
18 #include "chrome/browser/tab_contents/tab_util.h"
19 #include "chrome/common/pref_names.h"
20 #include "chrome/grit/chromium_strings.h"
21 #include "content/public/browser/browser_thread.h"
22 #include "content/public/browser/content_browser_client.h"
23 #include "content/public/browser/web_contents.h"
24 #include "content/public/browser/web_contents_delegate.h"
25 #include "content/public/browser/web_contents_observer.h"
26 #include "grit/theme_resources.h"
27 #include "net/base/net_util.h"
28 #include "ui/base/l10n/l10n_util.h"
29 #include "ui/base/resource/resource_bundle.h"
30 #include "ui/gfx/image/image_skia.h"
32 #if defined(ENABLE_EXTENSIONS)
33 #include "chrome/common/extensions/extension_constants.h"
34 #include "extensions/browser/extension_registry.h"
35 #include "extensions/common/extension.h"
38 using content::BrowserThread
;
39 using content::WebContents
;
43 #if defined(ENABLE_EXTENSIONS)
44 const extensions::Extension
* GetExtension(WebContents
* web_contents
) {
45 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
50 extensions::ExtensionRegistry
* registry
=
51 extensions::ExtensionRegistry::Get(web_contents
->GetBrowserContext());
52 return registry
->enabled_extensions().GetExtensionOrAppByURL(
53 web_contents
->GetURL());
56 bool IsWhitelistedExtension(const extensions::Extension
* extension
) {
57 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
59 static const char* const kExtensionWhitelist
[] = {
60 extension_misc::kHotwordExtensionId
,
61 extension_misc::kHotwordNewExtensionId
,
64 for (size_t i
= 0; i
< arraysize(kExtensionWhitelist
); ++i
) {
65 if (extension
->id() == kExtensionWhitelist
[i
])
71 #endif // defined(ENABLE_EXTENSIONS)
73 // Gets the security originator of the tab. It returns a string with no '/'
74 // at the end to display in the UI.
75 base::string16
GetSecurityOrigin(WebContents
* web_contents
) {
76 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
79 return base::string16();
81 std::string security_origin
= web_contents
->GetURL().GetOrigin().spec();
83 // Remove the last character if it is a '/'.
84 if (!security_origin
.empty()) {
85 std::string::iterator it
= security_origin
.end() - 1;
87 security_origin
.erase(it
);
90 return base::UTF8ToUTF16(security_origin
);
93 base::string16
GetTitle(WebContents
* web_contents
) {
94 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
97 return base::string16();
99 #if defined(ENABLE_EXTENSIONS)
100 const extensions::Extension
* const extension
= GetExtension(web_contents
);
102 return base::UTF8ToUTF16(extension
->name());
105 base::string16 tab_title
= web_contents
->GetTitle();
107 if (tab_title
.empty()) {
108 // If the page's title is empty use its security originator.
109 tab_title
= GetSecurityOrigin(web_contents
);
111 // If the page's title matches its URL, use its security originator.
113 Profile::FromBrowserContext(web_contents
->GetBrowserContext());
114 std::string languages
=
115 profile
->GetPrefs()->GetString(prefs::kAcceptLanguages
);
116 if (tab_title
== net::FormatUrl(web_contents
->GetURL(), languages
))
117 tab_title
= GetSecurityOrigin(web_contents
);
125 // Stores usage counts for all the capture devices associated with a single
126 // WebContents instance. Instances of this class are owned by
127 // MediaStreamCaptureIndicator. They also observe for the destruction of their
128 // corresponding WebContents and trigger their own deletion from their
129 // MediaStreamCaptureIndicator.
130 class MediaStreamCaptureIndicator::WebContentsDeviceUsage
131 : public content::WebContentsObserver
{
133 WebContentsDeviceUsage(scoped_refptr
<MediaStreamCaptureIndicator
> indicator
,
134 WebContents
* web_contents
)
135 : WebContentsObserver(web_contents
),
136 indicator_(indicator
),
139 mirroring_ref_count_(0),
140 weak_factory_(this) {
143 bool IsCapturingAudio() const { return audio_ref_count_
> 0; }
144 bool IsCapturingVideo() const { return video_ref_count_
> 0; }
145 bool IsMirroring() const { return mirroring_ref_count_
> 0; }
147 scoped_ptr
<content::MediaStreamUI
> RegisterMediaStream(
148 const content::MediaStreamDevices
& devices
);
150 // Increment ref-counts up based on the type of each device provided.
151 void AddDevices(const content::MediaStreamDevices
& devices
);
153 // Decrement ref-counts up based on the type of each device provided.
154 void RemoveDevices(const content::MediaStreamDevices
& devices
);
157 // content::WebContentsObserver overrides.
158 void WebContentsDestroyed() override
{
159 indicator_
->UnregisterWebContents(web_contents());
162 scoped_refptr
<MediaStreamCaptureIndicator
> indicator_
;
163 int audio_ref_count_
;
164 int video_ref_count_
;
165 int mirroring_ref_count_
;
167 base::WeakPtrFactory
<WebContentsDeviceUsage
> weak_factory_
;
169 DISALLOW_COPY_AND_ASSIGN(WebContentsDeviceUsage
);
172 // Implements MediaStreamUI interface. Instances of this class are created for
173 // each MediaStream and their ownership is passed to MediaStream implementation
174 // in the content layer. Each UIDelegate keeps a weak pointer to the
175 // corresponding WebContentsDeviceUsage object to deliver updates about state of
177 class MediaStreamCaptureIndicator::UIDelegate
: public content::MediaStreamUI
{
179 UIDelegate(base::WeakPtr
<WebContentsDeviceUsage
> device_usage
,
180 const content::MediaStreamDevices
& devices
)
181 : device_usage_(device_usage
),
184 DCHECK(!devices_
.empty());
187 ~UIDelegate() override
{
188 if (started_
&& device_usage_
.get())
189 device_usage_
->RemoveDevices(devices_
);
193 // content::MediaStreamUI interface.
194 gfx::NativeViewId
OnStarted(const base::Closure
& close_callback
) override
{
197 if (device_usage_
.get())
198 device_usage_
->AddDevices(devices_
);
202 base::WeakPtr
<WebContentsDeviceUsage
> device_usage_
;
203 content::MediaStreamDevices devices_
;
206 DISALLOW_COPY_AND_ASSIGN(UIDelegate
);
210 scoped_ptr
<content::MediaStreamUI
>
211 MediaStreamCaptureIndicator::WebContentsDeviceUsage::RegisterMediaStream(
212 const content::MediaStreamDevices
& devices
) {
213 return make_scoped_ptr(new UIDelegate( weak_factory_
.GetWeakPtr(), devices
));
216 void MediaStreamCaptureIndicator::WebContentsDeviceUsage::AddDevices(
217 const content::MediaStreamDevices
& devices
) {
218 for (content::MediaStreamDevices::const_iterator it
= devices
.begin();
219 it
!= devices
.end(); ++it
) {
220 if (it
->type
== content::MEDIA_TAB_AUDIO_CAPTURE
||
221 it
->type
== content::MEDIA_TAB_VIDEO_CAPTURE
) {
222 ++mirroring_ref_count_
;
223 } else if (content::IsAudioInputMediaType(it
->type
)) {
225 } else if (content::IsVideoMediaType(it
->type
)) {
233 web_contents()->NotifyNavigationStateChanged(content::INVALIDATE_TYPE_TAB
);
235 indicator_
->UpdateNotificationUserInterface();
238 void MediaStreamCaptureIndicator::WebContentsDeviceUsage::RemoveDevices(
239 const content::MediaStreamDevices
& devices
) {
240 for (content::MediaStreamDevices::const_iterator it
= devices
.begin();
241 it
!= devices
.end(); ++it
) {
242 if (it
->type
== content::MEDIA_TAB_AUDIO_CAPTURE
||
243 it
->type
== content::MEDIA_TAB_VIDEO_CAPTURE
) {
244 --mirroring_ref_count_
;
245 } else if (content::IsAudioInputMediaType(it
->type
)) {
247 } else if (content::IsVideoMediaType(it
->type
)) {
254 DCHECK_GE(audio_ref_count_
, 0);
255 DCHECK_GE(video_ref_count_
, 0);
256 DCHECK_GE(mirroring_ref_count_
, 0);
258 web_contents()->NotifyNavigationStateChanged(content::INVALIDATE_TYPE_TAB
);
259 indicator_
->UpdateNotificationUserInterface();
262 MediaStreamCaptureIndicator::MediaStreamCaptureIndicator()
263 : status_icon_(NULL
),
265 camera_image_(NULL
) {
268 MediaStreamCaptureIndicator::~MediaStreamCaptureIndicator() {
269 // The user is responsible for cleaning up by reporting the closure of any
270 // opened devices. However, there exists a race condition at shutdown: The UI
271 // thread may be stopped before CaptureDevicesClosed() posts the task to
272 // invoke DoDevicesClosedOnUIThread(). In this case, usage_map_ won't be
273 // empty like it should.
274 DCHECK(usage_map_
.empty() ||
275 !BrowserThread::IsMessageLoopValid(BrowserThread::UI
));
278 scoped_ptr
<content::MediaStreamUI
>
279 MediaStreamCaptureIndicator::RegisterMediaStream(
280 content::WebContents
* web_contents
,
281 const content::MediaStreamDevices
& devices
) {
282 WebContentsDeviceUsage
* usage
= usage_map_
.get(web_contents
);
284 usage
= new WebContentsDeviceUsage(this, web_contents
);
285 usage_map_
.add(web_contents
, make_scoped_ptr(usage
));
287 return usage
->RegisterMediaStream(devices
);
290 void MediaStreamCaptureIndicator::ExecuteCommand(int command_id
,
292 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
295 command_id
- IDC_MEDIA_CONTEXT_MEDIA_STREAM_CAPTURE_LIST_FIRST
;
297 DCHECK_GT(static_cast<int>(command_targets_
.size()), index
);
298 WebContents
* web_contents
= command_targets_
[index
];
299 if (ContainsKey(usage_map_
, web_contents
))
300 web_contents
->GetDelegate()->ActivateContents(web_contents
);
303 bool MediaStreamCaptureIndicator::IsCapturingUserMedia(
304 content::WebContents
* web_contents
) const {
305 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
307 WebContentsDeviceUsage
* usage
= usage_map_
.get(web_contents
);
308 return usage
&& (usage
->IsCapturingAudio() || usage
->IsCapturingVideo());
311 bool MediaStreamCaptureIndicator::IsCapturingVideo(
312 content::WebContents
* web_contents
) const {
313 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
315 WebContentsDeviceUsage
* usage
= usage_map_
.get(web_contents
);
316 return usage
&& usage
->IsCapturingVideo();
319 bool MediaStreamCaptureIndicator::IsCapturingAudio(
320 content::WebContents
* web_contents
) const {
321 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
323 WebContentsDeviceUsage
* usage
= usage_map_
.get(web_contents
);
324 return usage
&& usage
->IsCapturingAudio();
327 bool MediaStreamCaptureIndicator::IsBeingMirrored(
328 content::WebContents
* web_contents
) const {
329 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
331 WebContentsDeviceUsage
* usage
= usage_map_
.get(web_contents
);
332 return usage
&& usage
->IsMirroring();
335 void MediaStreamCaptureIndicator::UnregisterWebContents(
336 WebContents
* web_contents
) {
337 usage_map_
.erase(web_contents
);
338 UpdateNotificationUserInterface();
341 void MediaStreamCaptureIndicator::MaybeCreateStatusTrayIcon(bool audio
,
343 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
348 // If there is no browser process, we should not create the status tray.
349 if (!g_browser_process
)
352 StatusTray
* status_tray
= g_browser_process
->status_tray();
356 EnsureStatusTrayIconResources();
358 gfx::ImageSkia image
;
359 base::string16 tool_tip
;
360 GetStatusTrayIconInfo(audio
, video
, &image
, &tool_tip
);
361 DCHECK(!image
.isNull());
362 DCHECK(!tool_tip
.empty());
364 status_icon_
= status_tray
->CreateStatusIcon(
365 StatusTray::MEDIA_STREAM_CAPTURE_ICON
, image
, tool_tip
);
368 void MediaStreamCaptureIndicator::EnsureStatusTrayIconResources() {
369 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
372 mic_image_
= ResourceBundle::GetSharedInstance().GetImageSkiaNamed(
373 IDR_INFOBAR_MEDIA_STREAM_MIC
);
375 if (!camera_image_
) {
376 camera_image_
= ResourceBundle::GetSharedInstance().GetImageSkiaNamed(
377 IDR_INFOBAR_MEDIA_STREAM_CAMERA
);
380 DCHECK(camera_image_
);
383 void MediaStreamCaptureIndicator::MaybeDestroyStatusTrayIcon() {
384 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
389 // If there is no browser process, we should not do anything.
390 if (!g_browser_process
)
393 StatusTray
* status_tray
= g_browser_process
->status_tray();
394 if (status_tray
!= NULL
) {
395 status_tray
->RemoveStatusIcon(status_icon_
);
400 void MediaStreamCaptureIndicator::UpdateNotificationUserInterface() {
401 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
403 scoped_ptr
<StatusIconMenuModel
> menu(new StatusIconMenuModel(this));
406 int command_id
= IDC_MEDIA_CONTEXT_MEDIA_STREAM_CAPTURE_LIST_FIRST
;
407 command_targets_
.clear();
409 for (const auto& it
: usage_map_
) {
410 // Check if any audio and video devices have been used.
411 const WebContentsDeviceUsage
& usage
= *it
.second
;
412 if (!usage
.IsCapturingAudio() && !usage
.IsCapturingVideo())
415 WebContents
* const web_contents
= it
.first
;
417 // The audio/video icon is shown only for non-whitelisted extensions or on
418 // Android. For regular tabs on desktop, we show an indicator in the tab
420 #if defined(ENABLE_EXTENSIONS)
421 const extensions::Extension
* extension
= GetExtension(web_contents
);
422 if (!extension
|| IsWhitelistedExtension(extension
))
426 audio
= audio
|| usage
.IsCapturingAudio();
427 video
= video
|| usage
.IsCapturingVideo();
429 command_targets_
.push_back(web_contents
);
430 menu
->AddItem(command_id
, GetTitle(web_contents
));
432 // If the menu item is not a label, enable it.
433 menu
->SetCommandIdEnabled(command_id
, command_id
!= IDC_MinimumLabelValue
);
435 // If reaching the maximum number, no more item will be added to the menu.
436 if (command_id
== IDC_MEDIA_CONTEXT_MEDIA_STREAM_CAPTURE_LIST_LAST
)
441 if (command_targets_
.empty()) {
442 MaybeDestroyStatusTrayIcon();
446 // The icon will take the ownership of the passed context menu.
447 MaybeCreateStatusTrayIcon(audio
, video
);
449 status_icon_
->SetContextMenu(menu
.Pass());
453 void MediaStreamCaptureIndicator::GetStatusTrayIconInfo(
456 gfx::ImageSkia
* image
,
457 base::string16
* tool_tip
) {
458 DCHECK_CURRENTLY_ON(BrowserThread::UI
);
459 DCHECK(audio
|| video
);
464 if (audio
&& video
) {
465 message_id
= IDS_MEDIA_STREAM_STATUS_TRAY_TEXT_AUDIO_AND_VIDEO
;
466 *image
= *camera_image_
;
467 } else if (audio
&& !video
) {
468 message_id
= IDS_MEDIA_STREAM_STATUS_TRAY_TEXT_AUDIO_ONLY
;
469 *image
= *mic_image_
;
470 } else if (!audio
&& video
) {
471 message_id
= IDS_MEDIA_STREAM_STATUS_TRAY_TEXT_VIDEO_ONLY
;
472 *image
= *camera_image_
;
475 *tool_tip
= l10n_util::GetStringUTF16(message_id
);