From c73c9d3ae844c27ea74c2a4443d020c66459c881 Mon Sep 17 00:00:00 2001 From: fujunwei Date: Tue, 17 May 2016 09:18:27 +0800 Subject: [PATCH] [Android] Support third part media player on Crosswalk A requirement from an important customer who want to forward web resources with proxy on Crosswalk, but android system MediaPlayer can't set a proxy with a standard API. The ExoMediaPlayer is playing videos and music is a popular activity on Android devices, and it can be configured with proxy. https://developer.android.com/guide/topics/media/exoplayer.html BUG=XWALK-6770 --- .../chromium/media/ExternalMediaPlayer.java | 123 ++++++++++++++++++ .../org/chromium/media/MediaPlayerBridge.java | 25 ++-- 2 files changed, 139 insertions(+), 9 deletions(-) create mode 100644 media/base/android/java/src/org/chromium/media/ExternalMediaPlayer.java diff --git a/media/base/android/java/src/org/chromium/media/ExternalMediaPlayer.java b/media/base/android/java/src/org/chromium/media/ExternalMediaPlayer.java new file mode 100644 index 0000000000000..088c494b7693b --- /dev/null +++ b/media/base/android/java/src/org/chromium/media/ExternalMediaPlayer.java @@ -0,0 +1,123 @@ +// 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. + +package org.chromium.media; + +import android.content.Context; +import android.media.MediaPlayer; +import android.media.MediaPlayer.TrackInfo; +import android.net.Uri; +import android.view.Surface; + +import org.chromium.base.Log; + +import java.io.FileDescriptor; +import java.io.IOException; +import java.util.Map; + +/** +* Use Android system MediaPlayer by default. +*/ +public class ExternalMediaPlayer { + private static final String TAG = "ExternalMediaPlayer"; + + private MediaPlayer mPlayer; + + protected MediaPlayer getLocalPlayer() { + if (mPlayer == null) { + mPlayer = new MediaPlayer(); + Log.d(TAG, "Create an Android System Media Player."); + } + return mPlayer; + } + + public void setSurface(Surface surface) { + getLocalPlayer().setSurface(surface); + } + + public void setDataSource(Context context, Uri uri, Map headers) + throws Exception { + getLocalPlayer().setDataSource(context, uri, headers); + } + + public void setDataSource(FileDescriptor fd, long offset, long length) throws IOException { + getLocalPlayer().setDataSource(fd, offset, length); + } + + public void setDataSource(Context context, Uri uri) throws IOException { + getLocalPlayer().setDataSource(context, uri); + } + + public void prepareAsync() throws IllegalStateException { + getLocalPlayer().prepareAsync(); + } + + public TrackInfo[] getTrackInfo() throws RuntimeException { + return getLocalPlayer().getTrackInfo(); + } + + public boolean isPlaying() { + return getLocalPlayer().isPlaying(); + } + + public int getVideoWidth() { + return getLocalPlayer().getVideoWidth(); + } + + public int getVideoHeight() { + return getLocalPlayer().getVideoHeight(); + } + + public int getCurrentPosition() { + return getLocalPlayer().getCurrentPosition(); + } + + public int getDuration() { + return getLocalPlayer().getDuration(); + } + + public void release() { + getLocalPlayer().release(); + } + + public void setVolume(float volume1, float volume2) { + getLocalPlayer().setVolume(volume1, volume2); + } + + public void start() { + getLocalPlayer().start(); + } + + public void pause() { + getLocalPlayer().pause(); + } + + public void seekTo(int msec) { + getLocalPlayer().seekTo(msec); + } + + public void setOnBufferingUpdateListener(MediaPlayer.OnBufferingUpdateListener listener) { + getLocalPlayer().setOnBufferingUpdateListener(listener); + } + + public void setOnCompletionListener(MediaPlayer.OnCompletionListener listener) { + getLocalPlayer().setOnCompletionListener(listener); + } + + public void setOnErrorListener(MediaPlayer.OnErrorListener listener) { + getLocalPlayer().setOnErrorListener(listener); + } + + public void setOnPreparedListener(MediaPlayer.OnPreparedListener listener) { + getLocalPlayer().setOnPreparedListener(listener); + } + + public void setOnSeekCompleteListener(MediaPlayer.OnSeekCompleteListener listener) { + getLocalPlayer().setOnSeekCompleteListener(listener); + } + + public void setOnVideoSizeChangedListener(MediaPlayer.OnVideoSizeChangedListener listener) { + getLocalPlayer().setOnVideoSizeChangedListener(listener); + } +} diff --git a/media/base/android/java/src/org/chromium/media/MediaPlayerBridge.java b/media/base/android/java/src/org/chromium/media/MediaPlayerBridge.java index 830cc20049210..667c0c8617c4e 100644 --- a/media/base/android/java/src/org/chromium/media/MediaPlayerBridge.java +++ b/media/base/android/java/src/org/chromium/media/MediaPlayerBridge.java @@ -35,12 +35,18 @@ */ @JNINamespace("media") public class MediaPlayerBridge { - + /** + * Give the host application a chance to take over MeidaPlayer. + */ public static class ResourceLoadingFilter { public boolean shouldOverrideResourceLoading( - MediaPlayer mediaPlayer, Context context, Uri uri) { + ExternalMediaPlayer mediaPlayer, Context context, Uri uri) { return false; } + + public ExternalMediaPlayer getExternalMediaPlayer() { + return null; + } } private static ResourceLoadingFilter sResourceLoadFilter = null; @@ -54,7 +60,7 @@ public static void setResourceLoadingFilter(ResourceLoadingFilter filter) { // Local player to forward this to. We don't initialize it here since the subclass might not // want it. private LoadDataUriTask mLoadDataUriTask; - private MediaPlayer mPlayer; + private ExternalMediaPlayer mPlayer; private long mNativeMediaPlayerBridge; @CalledByNative @@ -78,9 +84,10 @@ protected void destroy() { mNativeMediaPlayerBridge = 0; } - protected MediaPlayer getLocalPlayer() { + protected ExternalMediaPlayer getLocalPlayer() { if (mPlayer == null) { - mPlayer = new MediaPlayer(); + mPlayer = sResourceLoadFilter.getExternalMediaPlayer(); + if (mPlayer == null) mPlayer = new ExternalMediaPlayer(); } return mPlayer; } @@ -202,9 +209,9 @@ protected boolean setDataSource( headersMap.put("allow-cross-domain-redirect", "false"); } try { - if (sResourceLoadFilter != null && - sResourceLoadFilter.shouldOverrideResourceLoading( - getLocalPlayer(), context, uri)) { + if (sResourceLoadFilter != null + && sResourceLoadFilter.shouldOverrideResourceLoading( + getLocalPlayer(), context, uri)) { return true; } getLocalPlayer().setDataSource(context, uri, headersMap); @@ -372,7 +379,7 @@ private boolean canSeekBackward() { */ @CalledByNative protected AllowedOperations getAllowedOperations() { - MediaPlayer player = getLocalPlayer(); + ExternalMediaPlayer player = getLocalPlayer(); boolean canPause = true; boolean canSeekForward = true; boolean canSeekBackward = true;