| Index: media/base/android/media_player_renderer.cc
|
| diff --git a/media/base/android/media_player_renderer.cc b/media/base/android/media_player_renderer.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..fbe74b1d36e0d7709b875c71e7b7bc46cd09666d
|
| --- /dev/null
|
| +++ b/media/base/android/media_player_renderer.cc
|
| @@ -0,0 +1,249 @@
|
| +// Copyright 2016 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "media/base/android/media_player_renderer.h"
|
| +
|
| +#include <memory>
|
| +
|
| +#include "content/browser/android/child_process_launcher_android.h"
|
| +#include "content/browser/media/android/media_resource_getter_impl.h"
|
| +#include "content/public/browser/browser_context.h"
|
| +#include "content/public/browser/render_process_host.h"
|
| +#include "content/public/browser/storage_partition.h"
|
| +#include "content/public/browser/web_contents.h"
|
| +#include "mojo/common/user_agent.h"
|
| +
|
| +namespace media {
|
| +
|
| +gl::ScopedJavaSurface MediaPlayerRenderer::surface_for_prototype;
|
| +
|
| +MediaPlayerRenderer::MediaPlayerRenderer(
|
| + content::RenderFrameHost* render_frame_host)
|
| + : surface_has_been_set_for_prototype(false),
|
| + render_frame_host_(render_frame_host),
|
| + duration_(base::TimeDelta::FromMicroseconds(0)),
|
| + weak_factory_(this) {}
|
| +
|
| +MediaPlayerRenderer::~MediaPlayerRenderer() {}
|
| +
|
| +void MediaPlayerRenderer::Initialize(
|
| + DemuxerStreamProvider* demuxer_stream_provider,
|
| + RendererClient* client,
|
| + const PipelineStatusCB& init_cb) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + if (demuxer_stream_provider->GetType() == DemuxerStreamProvider::Type::URL) {
|
| + url_ = demuxer_stream_provider->GetUrl();
|
| + renderer_client_ = client;
|
| +
|
| + // TODO(tguilbert): Use appropriate parameters (instead of GURL).
|
| + const int kIrrelevantMediaPlayerId = 0;
|
| + media_player_.reset(new MediaPlayerBridge(
|
| + kIrrelevantMediaPlayerId, url_, GURL(),
|
| + // TODO(tguilbert): What is the appropriate user agent to use?
|
| + mojo::common::GetUserAgent(), false, this,
|
| + base::Bind(&MediaPlayerRenderer::OnDecoderResourcesReleased,
|
| + weak_factory_.GetWeakPtr()),
|
| + GURL(), false, 0));
|
| +
|
| + // TODO(tguilbert): Register and Send the proper surface ID. See
|
| + // crbug.com/627658
|
| + SetSurfaceInternal_ForPrototype();
|
| +
|
| + // media_player_->SetVideoSurface(content::GetViewSurface(surface_id_));
|
| + media_player_->Initialize();
|
| + init_cb.Run(PIPELINE_OK);
|
| + } else {
|
| + LOG(ERROR) << "DemuxerStreamProvider is not of Type URL";
|
| + init_cb.Run(PIPELINE_ERROR_INITIALIZATION_FAILED);
|
| + }
|
| +}
|
| +
|
| +void MediaPlayerRenderer::SetSurfaceInternal_ForPrototype() {
|
| + if (!surface_for_prototype.IsEmpty() && !surface_has_been_set_for_prototype) {
|
| + media_player_->SetVideoSurface(
|
| + std::move(MediaPlayerRenderer::surface_for_prototype));
|
| + surface_has_been_set_for_prototype = true;
|
| + }
|
| +}
|
| +
|
| +void MediaPlayerRenderer::SetCdm(CdmContext* cdm_context,
|
| + const CdmAttachedCB& cdm_attached_cb) {
|
| + DVLOG(2) << __FUNCTION__;
|
| + NOTREACHED();
|
| +}
|
| +
|
| +void MediaPlayerRenderer::SetSurfaceId(int surface_id) {
|
| + surface_id_ = surface_id;
|
| +}
|
| +
|
| +void MediaPlayerRenderer::Flush(const base::Closure& flush_cb) {
|
| + DVLOG(2) << __FUNCTION__;
|
| + flush_cb.Run();
|
| +}
|
| +
|
| +void MediaPlayerRenderer::StartPlayingFrom(base::TimeDelta time) {
|
| + DVLOG(2) << __FUNCTION__;
|
| + // TODO(tguilbert): MediaPlayer's Start() is idempotent, but investigate
|
| + // MediaPlayerBridge's idempotency when it is in an error state. (It seems
|
| + // that calling start after receiving an error is counted as a new error).
|
| + //
|
| + // TODO(tguilbert): make sure IsPlayerReady() is true, and handle the cases
|
| + // when it's not.
|
| +
|
| + media_player_->Start();
|
| + media_player_->SeekTo(time);
|
| +}
|
| +
|
| +void MediaPlayerRenderer::SetPlaybackRate(double playback_rate) {
|
| + DVLOG(2) << __FUNCTION__;
|
| + // TODO(tguilbert): Should the state be save in order to prevent redundant
|
| + // calls?
|
| + if (playback_rate == 0) {
|
| + media_player_->Pause(false);
|
| + } else {
|
| + //
|
| + // TODO(tguilbert): Make sure IsPlayerReady() is true, and handle the cases
|
| + // when it's not.
|
| + media_player_->Start();
|
| + }
|
| +}
|
| +
|
| +void MediaPlayerRenderer::SetVolume(float volume) {
|
| + media_player_->SetVolume(volume);
|
| +}
|
| +
|
| +base::TimeDelta MediaPlayerRenderer::GetMediaTime() {
|
| + return media_player_->GetCurrentTime();
|
| +}
|
| +
|
| +bool MediaPlayerRenderer::HasAudio() {
|
| + return media_player_->HasAudio();
|
| +}
|
| +
|
| +bool MediaPlayerRenderer::HasVideo() {
|
| + return media_player_->HasVideo();
|
| +}
|
| +
|
| +MediaResourceGetter* MediaPlayerRenderer::GetMediaResourceGetter() {
|
| + DVLOG(1) << __FUNCTION__;
|
| +
|
| + if (!media_resource_getter_.get()) {
|
| + content::WebContents* web_contents =
|
| + content::WebContents::FromRenderFrameHost(render_frame_host_);
|
| + content::RenderProcessHost* host = web_contents->GetRenderProcessHost();
|
| + content::BrowserContext* context = host->GetBrowserContext();
|
| + content::StoragePartition* partition = host->GetStoragePartition();
|
| + storage::FileSystemContext* file_system_context =
|
| + partition ? partition->GetFileSystemContext() : NULL;
|
| + // Eventually this needs to be fixed to pass the correct frame rather
|
| + // than just using the main frame.
|
| + media_resource_getter_.reset(new content::MediaResourceGetterImpl(
|
| + context, file_system_context, host->GetID(),
|
| + web_contents->GetMainFrame()->GetRoutingID()));
|
| + }
|
| + return media_resource_getter_.get();
|
| +}
|
| +
|
| +MediaUrlInterceptor* MediaPlayerRenderer::GetMediaUrlInterceptor() {
|
| + DVLOG(1) << __FUNCTION__;
|
| + return nullptr;
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnTimeUpdate(int player_id,
|
| + base::TimeDelta current_timestamp,
|
| + base::TimeTicks current_time_ticks) {
|
| + // TODO(tguilbert): Handle this if necessary.
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnMediaMetadataChanged(int player_id,
|
| + base::TimeDelta duration,
|
| + int width,
|
| + int height,
|
| + bool success) {
|
| + DVLOG(1) << __FUNCTION__ << " { duration: " << duration.InMilliseconds()
|
| + << "(ms) witdh: " << width << ", height: " << height
|
| + << ", success: " << success << "}";
|
| + if (duration_ != duration) {
|
| + duration_ = duration;
|
| + renderer_client_->OnDurationChange(duration);
|
| + }
|
| + if (!surface_has_been_set_for_prototype) {
|
| + SetSurfaceInternal_ForPrototype();
|
| + }
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnPlaybackComplete(int player_id) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + renderer_client_->OnEnded();
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnMediaInterrupted(int player_id) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + // TODO(tguilbert): Handle this if necessary.
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnBufferingUpdate(int player_id, int percentage) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + // TODO(tguilbert): Determine proper threshold for triggering the have enough,
|
| + // and only trigger it once.
|
| + renderer_client_->OnBufferingStateChange(BUFFERING_HAVE_ENOUGH);
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnSeekComplete(int player_id,
|
| + const base::TimeDelta& current_time) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + // TODO(tguilbert): Handle this if necessary.
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnError(int player_id, int error) {
|
| + DVLOG(1) << __FUNCTION__ << " Error: " << error;
|
| + // TODO(tguilbert): Use more detailed errors if needed.
|
| + //
|
| + // TODO(tguilbert): Save the error state and properly error out or reset on
|
| + // future calls.
|
| +
|
| + // Some errors are forwarded to the MediaPlayerListener, but are of no
|
| + // importance to us. Ignore these errors, which are reported as error 0 by
|
| + // MediaPlayerListener.
|
| + if (error)
|
| + renderer_client_->OnError(PIPELINE_ERROR_COULD_NOT_RENDER);
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnVideoSizeChanged(int player_id,
|
| + int width,
|
| + int height) {
|
| + DVLOG(2) << __FUNCTION__;
|
| + renderer_client_->OnVideoNaturalSizeChange(gfx::Size(width, height));
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnWaitingForDecryptionKey(int player_id) {
|
| + DVLOG(1) << __FUNCTION__;
|
| + NOTREACHED();
|
| +}
|
| +
|
| +MediaPlayerAndroid* MediaPlayerRenderer::GetFullscreenPlayer() {
|
| + NOTREACHED();
|
| + return nullptr;
|
| +}
|
| +
|
| +MediaPlayerAndroid* MediaPlayerRenderer::GetPlayer(int player_id) {
|
| + NOTREACHED();
|
| + return nullptr;
|
| +}
|
| +
|
| +bool MediaPlayerRenderer::RequestPlay(int player_id,
|
| + base::TimeDelta duration,
|
| + bool has_audio) {
|
| + // TODO(tguilbert): Determine wheter or not this function should ever
|
| + // return false.
|
| + return true;
|
| +}
|
| +
|
| +void MediaPlayerRenderer::OnDecoderResourcesReleased(int player_id) {
|
| + // TODO(tguilbert): Since we are not using a pool of MediaPlayerAndroid
|
| + // instances, this function is not relevant. Investigate whether the use of
|
| + // the MediaThrottler is needed.
|
| +}
|
| +
|
| +} // namespace media
|
|
|