Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(18)

Side by Side Diff: content/browser/media/android/media_web_contents_observer_android.cc

Issue 1570043002: Implement MediaSession on top of the WebMediaPlayerDelegate. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@media_session
Patch Set: Rebase yet again. Created 4 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2016 The Chromium Authors. All rights reserved. 1 // Copyright 2016 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "content/browser/media/android/media_web_contents_observer_android.h" 5 #include "content/browser/media/android/media_web_contents_observer_android.h"
6 6
7 #include "content/browser/media/android/browser_media_player_manager.h" 7 #include "content/browser/media/android/browser_media_player_manager.h"
8 #include "content/browser/media/android/browser_media_session_manager.h" 8 #include "content/browser/media/android/browser_media_session_manager.h"
9 #include "content/browser/media/android/media_session.h"
10 #include "content/browser/media/android/media_session_controller.h"
9 #include "content/browser/media/android/media_session_observer.h" 11 #include "content/browser/media/android/media_session_observer.h"
10 #include "content/browser/media/cdm/browser_cdm_manager.h" 12 #include "content/browser/media/cdm/browser_cdm_manager.h"
11 #include "content/browser/web_contents/web_contents_impl.h" 13 #include "content/browser/web_contents/web_contents_impl.h"
14 #include "content/common/frame_messages.h"
12 #include "content/common/media/media_player_messages_android.h" 15 #include "content/common/media/media_player_messages_android.h"
13 #include "content/common/media/media_session_messages_android.h" 16 #include "content/common/media/media_session_messages_android.h"
14 #include "content/public/browser/render_frame_host.h" 17 #include "content/public/browser/render_frame_host.h"
15 #include "content/public/browser/web_contents.h" 18 #include "content/public/browser/web_contents.h"
16 #include "ipc/ipc_message_macros.h" 19 #include "ipc/ipc_message_macros.h"
17 #include "media/base/android/media_player_android.h" 20 #include "media/base/android/media_player_android.h"
18 21
19 namespace content { 22 namespace content {
20 23
21 MediaWebContentsObserverAndroid::MediaWebContentsObserverAndroid( 24 MediaWebContentsObserverAndroid::MediaWebContentsObserverAndroid(
(...skipping 29 matching lines...) Expand all
51 auto it = media_session_managers_.find(render_frame_host); 54 auto it = media_session_managers_.find(render_frame_host);
52 if (it != media_session_managers_.end()) 55 if (it != media_session_managers_.end())
53 return it->second; 56 return it->second;
54 57
55 BrowserMediaSessionManager* manager = 58 BrowserMediaSessionManager* manager =
56 new BrowserMediaSessionManager(render_frame_host); 59 new BrowserMediaSessionManager(render_frame_host);
57 media_session_managers_.set(render_frame_host, make_scoped_ptr(manager)); 60 media_session_managers_.set(render_frame_host, make_scoped_ptr(manager));
58 return manager; 61 return manager;
59 } 62 }
60 63
64 bool MediaWebContentsObserverAndroid::RequestPlay(
65 RenderFrameHost* render_frame_host,
66 int64_t player_cookie,
67 bool has_audio,
68 bool is_remote,
69 base::TimeDelta duration) {
70 const MediaPlayerId id(render_frame_host, player_cookie);
71
72 // Remove any existing entry if it has a different audio parameter; when WMPA
73 // is used, the value of |has_audio| may not be initially known.
74 auto it = media_session_map_.find(id);
75 if (it != media_session_map_.end() && it->second->has_audio() != has_audio)
76 media_session_map_.erase(it);
77
78 // |has_video| forced to true since the value doesn't matter at present.
79 OnMediaPlayingNotification(render_frame_host, player_cookie, true, has_audio,
80 is_remote, duration);
81 return media_session_map_.find(id) != media_session_map_.end();
82 }
83
61 #if defined(VIDEO_HOLE) 84 #if defined(VIDEO_HOLE)
62 void MediaWebContentsObserverAndroid::OnFrameInfoUpdated() { 85 void MediaWebContentsObserverAndroid::OnFrameInfoUpdated() {
63 for (auto it = media_player_managers_.begin(); 86 for (auto it = media_player_managers_.begin();
64 it != media_player_managers_.end(); ++it) { 87 it != media_player_managers_.end(); ++it) {
65 it->second->OnFrameInfoUpdated(); 88 it->second->OnFrameInfoUpdated();
66 } 89 }
67 } 90 }
68 #endif // defined(VIDEO_HOLE) 91 #endif // defined(VIDEO_HOLE)
69 92
70 void MediaWebContentsObserverAndroid::RenderFrameDeleted( 93 void MediaWebContentsObserverAndroid::RenderFrameDeleted(
71 RenderFrameHost* render_frame_host) { 94 RenderFrameHost* render_frame_host) {
72 MediaWebContentsObserver::RenderFrameDeleted(render_frame_host); 95 MediaWebContentsObserver::RenderFrameDeleted(render_frame_host);
73 96
97 for (auto it = media_session_map_.begin(); it != media_session_map_.end();) {
98 if (it->first.first == render_frame_host)
99 it = media_session_map_.erase(it);
100 else
101 ++it;
102 }
103
74 // Always destroy the media players before CDMs because we do not support 104 // Always destroy the media players before CDMs because we do not support
75 // detaching CDMs from media players yet. See http://crbug.com/330324 105 // detaching CDMs from media players yet. See http://crbug.com/330324
76 media_player_managers_.erase(render_frame_host); 106 media_player_managers_.erase(render_frame_host);
77 media_session_managers_.erase(render_frame_host); 107 media_session_managers_.erase(render_frame_host);
78 108
79 // TODO(xhwang): Currently MediaWebContentsObserver, BrowserMediaPlayerManager 109 // TODO(xhwang): Currently MediaWebContentsObserver, BrowserMediaPlayerManager
80 // and BrowserCdmManager all run on browser UI thread. So this call is okay. 110 // and BrowserCdmManager all run on browser UI thread. So this call is okay.
81 // In the future we need to support the case where MediaWebContentsObserver 111 // In the future we need to support the case where MediaWebContentsObserver
82 // get notified on browser UI thread, but BrowserMediaPlayerManager and 112 // get notified on browser UI thread, but BrowserMediaPlayerManager and
83 // BrowserCdmManager run on a different thread. 113 // BrowserCdmManager run on a different thread.
84 BrowserCdmManager* browser_cdm_manager = 114 BrowserCdmManager* browser_cdm_manager =
85 BrowserCdmManager::FromProcess(render_frame_host->GetProcess()->GetID()); 115 BrowserCdmManager::FromProcess(render_frame_host->GetProcess()->GetID());
86 if (browser_cdm_manager) 116 if (browser_cdm_manager)
87 browser_cdm_manager->RenderFrameDeleted(render_frame_host->GetRoutingID()); 117 browser_cdm_manager->RenderFrameDeleted(render_frame_host->GetRoutingID());
88 } 118 }
89 119
90 bool MediaWebContentsObserverAndroid::OnMessageReceived( 120 bool MediaWebContentsObserverAndroid::OnMessageReceived(
91 const IPC::Message& msg, 121 const IPC::Message& msg,
92 RenderFrameHost* render_frame_host) { 122 RenderFrameHost* render_frame_host) {
123 // Receive play/pause/destroyed messages, but don't mark as processed so they
124 // are also handled by MediaWebContentsObserver.
125 OnMediaPlayerDelegateMessageReceived(msg, render_frame_host);
126
93 if (MediaWebContentsObserver::OnMessageReceived(msg, render_frame_host)) 127 if (MediaWebContentsObserver::OnMessageReceived(msg, render_frame_host))
94 return true; 128 return true;
95 129
96 if (OnMediaPlayerMessageReceived(msg, render_frame_host)) 130 if (OnMediaPlayerMessageReceived(msg, render_frame_host))
97 return true; 131 return true;
98 132
99 return OnMediaPlayerSetCdmMessageReceived(msg, render_frame_host); 133 return OnMediaPlayerSetCdmMessageReceived(msg, render_frame_host);
100 } 134 }
101 135
136 void MediaWebContentsObserverAndroid::OnMediaPlayerDelegateMessageReceived(
137 const IPC::Message& msg,
138 RenderFrameHost* render_frame_host) {
139 // TODO(dalecurtis): These should no longer be FrameHostMsg.
nasko 2016/01/22 17:28:32 Well, you are adding some new messages. Why make t
DaleCurtis 2016/01/23 02:11:00 Figured it would grow the CL too much, but it's no
140 IPC_BEGIN_MESSAGE_MAP_WITH_PARAM(MediaWebContentsObserverAndroid, msg,
141 render_frame_host)
142 IPC_MESSAGE_HANDLER(FrameHostMsg_MediaDestroyedNotification,
143 OnMediaDestroyedNotification)
144 IPC_MESSAGE_HANDLER(FrameHostMsg_MediaPlayingNotification,
145 OnMediaPlayingNotification)
146 IPC_MESSAGE_HANDLER(FrameHostMsg_MediaPausedNotification,
147 OnMediaPausedNotification)
148 IPC_END_MESSAGE_MAP()
149 }
150
102 bool MediaWebContentsObserverAndroid::OnMediaPlayerMessageReceived( 151 bool MediaWebContentsObserverAndroid::OnMediaPlayerMessageReceived(
103 const IPC::Message& msg, 152 const IPC::Message& msg,
104 RenderFrameHost* render_frame_host) { 153 RenderFrameHost* render_frame_host) {
105 bool handled = true; 154 bool handled = true;
106 IPC_BEGIN_MESSAGE_MAP(MediaWebContentsObserverAndroid, msg) 155 IPC_BEGIN_MESSAGE_MAP(MediaWebContentsObserverAndroid, msg)
107 IPC_MESSAGE_FORWARD(MediaPlayerHostMsg_EnterFullscreen, 156 IPC_MESSAGE_FORWARD(MediaPlayerHostMsg_EnterFullscreen,
108 GetMediaPlayerManager(render_frame_host), 157 GetMediaPlayerManager(render_frame_host),
109 BrowserMediaPlayerManager::OnEnterFullscreen) 158 BrowserMediaPlayerManager::OnEnterFullscreen)
110 IPC_MESSAGE_FORWARD(MediaPlayerHostMsg_Initialize, 159 IPC_MESSAGE_FORWARD(MediaPlayerHostMsg_Initialize,
111 GetMediaPlayerManager(render_frame_host), 160 GetMediaPlayerManager(render_frame_host),
(...skipping 78 matching lines...) Expand 10 before | Expand all | Expand 10 after
190 if (!cdm) { 239 if (!cdm) {
191 NOTREACHED() << "OnSetCdm: CDM not found for " << cdm_id; 240 NOTREACHED() << "OnSetCdm: CDM not found for " << cdm_id;
192 return; 241 return;
193 } 242 }
194 243
195 // TODO(xhwang): This could possibly fail. In that case we should reject the 244 // TODO(xhwang): This could possibly fail. In that case we should reject the
196 // promise. 245 // promise.
197 media_player->SetCdm(cdm); 246 media_player->SetCdm(cdm);
198 } 247 }
199 248
249 void MediaWebContentsObserverAndroid::OnMediaDestroyedNotification(
250 RenderFrameHost* render_frame_host,
251 int64_t player_cookie) {
252 media_session_map_.erase(MediaPlayerId(render_frame_host, player_cookie));
253 }
254
255 void MediaWebContentsObserverAndroid::OnMediaPlayingNotification(
256 RenderFrameHost* render_frame_host,
257 int64_t player_cookie,
258 bool has_video,
259 bool has_audio,
260 bool is_remote,
261 base::TimeDelta duration) {
262 // Since we don't remove session instances on pause, there may be an existing
263 // instance for this playback attempt.
mlamouri (slow - plz ping) 2016/01/22 16:58:31 I think this is going to lead to bugs. If you play
DaleCurtis 2016/01/23 02:11:00 I think this actually makes things a bit more comp
264 const MediaPlayerId id(render_frame_host, player_cookie);
265 if (media_session_map_.find(id) != media_session_map_.end())
266 return;
267
268 scoped_ptr<MediaSessionController> controller(
269 new MediaSessionController(id, this));
270
271 // If initialize fails, the controller should be destroyed and a new one
272 // attempted later after another playback attempt occurs.
273 if (!controller->Initialize(has_audio, is_remote, duration))
274 return;
275
276 media_session_map_[id] = std::move(controller);
277 }
278
279 void MediaWebContentsObserverAndroid::OnMediaPausedNotification(
280 RenderFrameHost* render_frame_host,
281 int64_t player_cookie,
282 bool reached_end_of_stream) {
283 // Drop the session if playback completes normally.
284 if (reached_end_of_stream) {
285 OnMediaDestroyedNotification(render_frame_host, player_cookie);
286 return;
287 }
288
289 auto it =
290 media_session_map_.find(MediaPlayerId(render_frame_host, player_cookie));
291 if (it == media_session_map_.end())
292 return;
293
294 it->second->PausePlayback();
295 }
296
200 } // namespace content 297 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698