Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2013 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. | |
| 4 | |
| 5 #include "chrome/browser/media/android/remote/remote_media_player_manager.h" | |
| 6 | |
| 7 #include "chrome/browser/android/tab_android.h" | |
| 8 #include "chrome/common/chrome_content_client.h" | |
| 9 #include "content/common/media/media_player_messages_android.h" | |
| 10 #include "third_party/skia/include/core/SkBitmap.h" | |
| 11 #include "ui/gfx/android/java_bitmap.h" | |
| 12 | |
| 13 using media::MediaPlayerAndroid; | |
| 14 | |
| 15 namespace remote_media { | |
| 16 | |
| 17 RemoteMediaPlayerManager::RemoteMediaPlayerManager( | |
| 18 content::RenderFrameHost* render_frame_host, | |
| 19 content::MediaPlayersObserver* audio_monitor) | |
| 20 : BrowserMediaPlayerManager(render_frame_host, audio_monitor), | |
| 21 weak_ptr_factory_(this) { | |
| 22 } | |
| 23 | |
| 24 RemoteMediaPlayerManager::~RemoteMediaPlayerManager() {} | |
| 25 | |
| 26 void RemoteMediaPlayerManager::OnStart(int player_id) { | |
| 27 // TODO(aberent) This assumes this is the first time we have started this | |
| 28 // video, rather than restarting after pause. There is a lot of logic here | |
| 29 // that is unnecessary if we are restarting after pause. | |
| 30 if (MaybeStartPlayingRemotely(player_id)) | |
| 31 return; | |
| 32 | |
| 33 ReplaceRemotePlayerWithLocal(); | |
| 34 BrowserMediaPlayerManager::OnStart(player_id); | |
| 35 } | |
| 36 | |
| 37 void RemoteMediaPlayerManager::OnInitialize( | |
| 38 const MediaPlayerHostMsg_Initialize_Params& media_params) { | |
| 39 BrowserMediaPlayerManager::OnInitialize(media_params); | |
| 40 | |
| 41 MediaPlayerAndroid* player = GetPlayer(media_params.player_id); | |
| 42 if (player) { | |
| 43 CreateRemoteMediaPlayer(player); | |
| 44 RemoteMediaPlayerBridge* remote_player = GetRemotePlayer( | |
| 45 media_params.player_id); | |
| 46 if (remote_player) | |
| 47 remote_player->OnPlayerCreated(); | |
| 48 } | |
| 49 } | |
| 50 | |
| 51 void RemoteMediaPlayerManager::OnDestroyPlayer(int player_id) { | |
| 52 RemoteMediaPlayerBridge* player = GetRemotePlayer(player_id); | |
| 53 if (player) | |
| 54 player->OnPlayerDestroyed(); | |
| 55 BrowserMediaPlayerManager::OnDestroyPlayer(player_id); | |
| 56 } | |
| 57 | |
| 58 void RemoteMediaPlayerManager::OnReleaseResources(int player_id) { | |
| 59 // We only want to release resources of local players. | |
| 60 if (player_id != RemotePlayerId()) | |
| 61 BrowserMediaPlayerManager::OnReleaseResources(player_id); | |
| 62 } | |
| 63 | |
| 64 void RemoteMediaPlayerManager::OnRequestRemotePlayback(int player_id) { | |
| 65 RemoteMediaPlayerBridge* player = GetRemotePlayer(player_id); | |
| 66 if (player) | |
| 67 player->RequestRemotePlayback(); | |
| 68 } | |
| 69 | |
| 70 void RemoteMediaPlayerManager::OnRequestRemotePlaybackControl(int player_id) { | |
| 71 RemoteMediaPlayerBridge* player = GetRemotePlayer(player_id); | |
| 72 if (player) | |
| 73 player->RequestRemotePlaybackControl(); | |
| 74 } | |
| 75 | |
| 76 int RemoteMediaPlayerManager::GetTabId() { | |
| 77 if (!web_contents()) | |
| 78 return -1; | |
| 79 | |
| 80 TabAndroid* tab = TabAndroid::FromWebContents(web_contents()); | |
| 81 if (!tab) | |
| 82 return -1; | |
| 83 | |
| 84 return tab->GetAndroidId(); | |
| 85 } | |
| 86 | |
| 87 void RemoteMediaPlayerManager::OnSetPoster(int player_id, const GURL& url) { | |
| 88 RemoteMediaPlayerBridge* player = GetRemotePlayer(player_id); | |
| 89 | |
| 90 if (player && url.is_empty()) { | |
| 91 player->SetPosterBitmap(std::vector<SkBitmap>()); | |
| 92 } else { | |
| 93 // TODO(aberent) OnSetPoster is called when the attributes of the video | |
| 94 // element are parsed, which may be before OnInitialize is called. We are | |
| 95 // here relying on the image fetch taking longer than the delay until | |
| 96 // OnInitialize is called, and hence the player is created. This is not | |
| 97 // guaranteed. | |
| 98 content::WebContents::ImageDownloadCallback callback = base::Bind( | |
| 99 &RemoteMediaPlayerManager::DidDownloadPoster, | |
| 100 weak_ptr_factory_.GetWeakPtr(), player_id); | |
| 101 web_contents()->DownloadImage( | |
| 102 url, | |
| 103 false, // is_favicon, false so that cookies will be used. | |
| 104 0, // max_bitmap_size, 0 means no limit. | |
| 105 callback); | |
| 106 } | |
| 107 } | |
| 108 | |
| 109 void RemoteMediaPlayerManager::DidDownloadPoster( | |
| 110 int player_id, | |
| 111 int id, | |
| 112 int http_status_code, | |
| 113 const GURL& image_url, | |
| 114 const std::vector<SkBitmap>& bitmaps, | |
| 115 const std::vector<gfx::Size>& original_bitmap_sizes) { | |
| 116 RemoteMediaPlayerBridge* player = GetRemotePlayer(player_id); | |
| 117 if (player) | |
| 118 player->SetPosterBitmap(bitmaps); | |
| 119 } | |
| 120 | |
| 121 RemoteMediaPlayerBridge* RemoteMediaPlayerManager::CreateRemoteMediaPlayer( | |
| 122 MediaPlayerAndroid* local_player) { | |
| 123 RemoteMediaPlayerBridge* player = new RemoteMediaPlayerBridge( | |
| 124 local_player, | |
| 125 GetUserAgent(), | |
| 126 false, | |
| 127 this); | |
| 128 remote_players_.push_back(player); | |
| 129 player->Initialize(); | |
| 130 return player; | |
| 131 } | |
| 132 | |
| 133 int RemoteMediaPlayerManager::RemotePlayerId() { | |
| 134 // The remote player is created with the same id as the corresponding local | |
| 135 // player. | |
| 136 if (replaced_local_player_.get()) { | |
|
qinmin
2015/03/20 19:35:46
single line if else statement, no need for {} here
aberent
2015/03/23 17:35:31
Done.
| |
| 137 return replaced_local_player_->player_id(); | |
| 138 } else { | |
| 139 return -1; | |
| 140 } | |
| 141 } | |
| 142 | |
| 143 void RemoteMediaPlayerManager::ReplaceLocalPlayerWithRemote( | |
| 144 MediaPlayerAndroid* player) { | |
| 145 if (!player) | |
| 146 return; | |
| 147 | |
| 148 int player_id = player->player_id(); | |
| 149 if (player_id == RemotePlayerId()) { | |
| 150 // The player is already remote. | |
| 151 return; | |
| 152 } | |
| 153 | |
| 154 // Before we replace the new remote player, put the old local player back | |
| 155 // in its place. | |
| 156 ReplaceRemotePlayerWithLocal(); | |
| 157 | |
| 158 // Pause the local player first before replacing it. This will allow the local | |
| 159 // player to reset its state, such as the PowerSaveBlocker. | |
| 160 // We have to pause locally as well as telling the renderer to pause, because | |
| 161 // by the time the renderer comes back to us telling us to pause we will have | |
| 162 // switched players. | |
| 163 player->Pause(true); | |
| 164 Send(new MediaPlayerMsg_DidMediaPlayerPause(RoutingID(), player_id)); | |
| 165 | |
| 166 // Find the remote player | |
| 167 for (auto it = remote_players_.begin(); it != remote_players_.end(); ++it) { | |
| 168 if ((*it)->player_id() == player_id) { | |
| 169 replaced_local_player_ = SwapPlayer(player_id, *it); | |
| 170 | |
| 171 // Seek to the previous player's position. | |
| 172 (*it)->SeekTo(player->GetCurrentTime()); | |
| 173 | |
| 174 // SwapPlayers takes ownership, so we have to remove the remote player | |
| 175 // from the vector. | |
| 176 remote_players_.weak_erase(it); | |
| 177 break; | |
| 178 } | |
| 179 } | |
| 180 } | |
| 181 | |
| 182 void RemoteMediaPlayerManager::ReplaceRemotePlayerWithLocal() { | |
| 183 int player_id = RemotePlayerId(); | |
| 184 if (player_id == -1) | |
| 185 return; | |
| 186 | |
| 187 Send(new MediaPlayerMsg_DidMediaPlayerPause(RoutingID(), player_id)); | |
| 188 Send(new MediaPlayerMsg_DisconnectedFromRemoteDevice(RoutingID(), player_id)); | |
| 189 | |
| 190 scoped_ptr<MediaPlayerAndroid> remote_player = | |
| 191 SwapPlayer(player_id, replaced_local_player_.release()); | |
| 192 if (remote_player) { | |
| 193 // Seek to the previous player's position. | |
| 194 GetPlayer(player_id)->SeekTo(remote_player->GetCurrentTime()); | |
| 195 | |
| 196 remote_player->Release(); | |
| 197 // Add the remote player back into the list | |
| 198 remote_players_.push_back( | |
| 199 static_cast<RemoteMediaPlayerBridge *>(remote_player.release())); | |
| 200 } | |
| 201 } | |
| 202 | |
| 203 bool RemoteMediaPlayerManager::MaybeStartPlayingRemotely(int player_id) { | |
| 204 MediaPlayerAndroid* player = GetPlayer(player_id); | |
| 205 if (!player) | |
| 206 return false; | |
| 207 | |
| 208 RemoteMediaPlayerBridge* remote_player = GetRemotePlayer(player_id); | |
| 209 | |
| 210 if (!remote_player) | |
| 211 return false; | |
| 212 | |
| 213 if (remote_player->IsMediaPlayableRemotely() && | |
| 214 remote_player->IsRemotePlaybackAvailable() && | |
| 215 remote_player->IsRemotePlaybackPreferredForFrame()) { | |
| 216 ReplaceLocalPlayerWithRemote(player); | |
| 217 | |
| 218 remote_player->SetNativePlayer(); | |
| 219 remote_player->Start(); | |
| 220 | |
| 221 Send(new MediaPlayerMsg_DidMediaPlayerPlay(RoutingID(), player_id)); | |
| 222 | |
| 223 Send(new MediaPlayerMsg_ConnectedToRemoteDevice( | |
| 224 RoutingID(), | |
| 225 player_id, | |
| 226 remote_player->GetCastingMessage())); | |
| 227 | |
| 228 return true; | |
| 229 } | |
| 230 | |
| 231 return false; | |
| 232 } | |
| 233 | |
| 234 void RemoteMediaPlayerManager::OnRemoteDeviceSelected(int player_id) { | |
| 235 | |
| 236 MediaPlayerAndroid* player = GetPlayer(player_id); | |
| 237 if (!player) | |
| 238 return; | |
| 239 | |
| 240 if (MaybeStartPlayingRemotely(player_id)) | |
| 241 return; | |
| 242 OnStart(player_id); | |
| 243 } | |
| 244 | |
| 245 void RemoteMediaPlayerManager::OnRemoteDeviceUnselected(int player_id) { | |
| 246 if (player_id == RemotePlayerId()) | |
| 247 ReplaceRemotePlayerWithLocal(); | |
| 248 } | |
| 249 | |
| 250 void RemoteMediaPlayerManager::OnRemotePlaybackFinished(int player_id) { | |
| 251 if (player_id == RemotePlayerId()) | |
| 252 ReplaceRemotePlayerWithLocal(); | |
| 253 } | |
| 254 | |
| 255 void RemoteMediaPlayerManager::OnRouteAvailabilityChanged( | |
| 256 int player_id, bool routes_available) { | |
| 257 Send( | |
| 258 new MediaPlayerMsg_RemoteRouteAvailabilityChanged(RoutingID(), player_id, | |
| 259 routes_available)); | |
| 260 } | |
| 261 | |
| 262 void RemoteMediaPlayerManager::ReleaseFullscreenPlayer( | |
| 263 MediaPlayerAndroid* player) { | |
| 264 // Release the original player's resources, not the current fullscreen player | |
| 265 // (which is the remote player). | |
| 266 if (replaced_local_player_.get()) | |
| 267 replaced_local_player_->Release(); | |
| 268 else | |
| 269 BrowserMediaPlayerManager::ReleaseFullscreenPlayer(player); | |
| 270 } | |
| 271 | |
| 272 void RemoteMediaPlayerManager::OnPlaying(int player_id) { | |
| 273 Send(new MediaPlayerMsg_DidMediaPlayerPlay(RoutingID(),player_id)); | |
| 274 } | |
| 275 | |
| 276 void RemoteMediaPlayerManager::OnPaused(int player_id) { | |
| 277 Send(new MediaPlayerMsg_DidMediaPlayerPause(RoutingID(),player_id)); | |
| 278 } | |
| 279 | |
| 280 RemoteMediaPlayerBridge* RemoteMediaPlayerManager::GetRemotePlayer( | |
| 281 int player_id) { | |
| 282 if (player_id == RemotePlayerId()) { | |
| 283 return static_cast<RemoteMediaPlayerBridge*>(GetPlayer(player_id)); | |
| 284 } else { | |
| 285 for (RemoteMediaPlayerBridge* player : remote_players_) { | |
| 286 if (player->player_id() == player_id) { | |
| 287 return player; | |
| 288 } | |
| 289 } | |
| 290 return nullptr; | |
| 291 } | |
| 292 } | |
| 293 | |
| 294 void RemoteMediaPlayerManager::OnMediaMetadataChanged(int player_id, | |
| 295 base::TimeDelta duration, | |
| 296 int width, | |
| 297 int height, | |
| 298 bool success) { | |
| 299 if (player_id == RemotePlayerId() && replaced_local_player_) { | |
| 300 Send(new MediaPlayerMsg_MediaMetadataChanged( | |
| 301 RoutingID(), player_id, duration, | |
| 302 replaced_local_player_->GetVideoWidth(), | |
| 303 replaced_local_player_->GetVideoHeight(), success)); | |
| 304 } else { | |
| 305 BrowserMediaPlayerManager::OnMediaMetadataChanged(player_id, duration, | |
| 306 width, height, success); | |
| 307 } | |
| 308 } | |
| 309 | |
| 310 } // namespace remote_media | |
| OLD | NEW |