OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 |
OLD | NEW |