am 9299c6cd: Ensure sample runs on Gingerbread. (DO NOT MERGE)

* commit '9299c6cdd8e5d084ef4c10e5b7f044e0ea590641':
  Ensure sample runs on Gingerbread.  (DO NOT MERGE)
This commit is contained in:
Jeff Brown
2013-08-22 20:28:01 -07:00
committed by Android Git Automerger
4 changed files with 466 additions and 348 deletions

View File

@@ -16,16 +16,14 @@
package com.example.android.supportv7.media;
import com.example.android.supportv7.R;
import android.content.Context;
import android.net.Uri;
import android.os.Build;
import android.os.Handler;
import android.util.Log;
import android.media.MediaPlayer;
import android.view.Surface;
import android.view.Gravity;
import android.graphics.SurfaceTexture;
import android.view.SurfaceHolder;
import java.io.IOException;
/**
@@ -37,7 +35,6 @@ public class MediaPlayerWrapper implements
MediaPlayer.OnCompletionListener,
MediaPlayer.OnErrorListener,
MediaPlayer.OnSeekCompleteListener,
OverlayDisplayWindow.OverlayWindowListener,
MediaSessionManager.Callback {
private static final String TAG = "MediaPlayerWrapper";
private static final boolean DEBUG = false;
@@ -54,6 +51,7 @@ public class MediaPlayerWrapper implements
private int mState = STATE_IDLE;
private Callback mCallback;
private Surface mSurface;
private SurfaceHolder mSurfaceHolder;
private int mSeekToPos;
public MediaPlayerWrapper(Context context) {
@@ -140,21 +138,16 @@ public class MediaPlayerWrapper implements
}
}
//OverlayDisplayWindow listeners
@Override
public void onWindowCreated(Surface surface) {
if (DEBUG) {
Log.d(TAG, "onWindowCreated");
}
public void setSurface(Surface surface) {
mSurface = surface;
mMediaPlayer.setSurface(surface);
mSurfaceHolder = null;
updateSurface();
}
@Override
public void onWindowDestroyed() {
if (DEBUG) {
Log.d(TAG, "onWindowDestroyed");
}
public void setSurface(SurfaceHolder surfaceHolder) {
mSurface = null;
mSurfaceHolder = surfaceHolder;
updateSurface();
}
//MediaPlayer Listeners
@@ -244,9 +237,7 @@ public class MediaPlayerWrapper implements
mMediaPlayer.setOnCompletionListener(this);
mMediaPlayer.setOnErrorListener(this);
mMediaPlayer.setOnSeekCompleteListener(this);
if (mSurface != null) {
mMediaPlayer.setSurface(mSurface);
}
updateSurface();
mState = STATE_IDLE;
mSeekToPos = 0;
}
@@ -265,10 +256,32 @@ public class MediaPlayerWrapper implements
}
}
private void updateSurface() {
if (mSurface != null) {
// The setSurface API does not exist until V14+.
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
ICSMediaPlayer.setSurface(mMediaPlayer, mSurface);
} else {
throw new UnsupportedOperationException("MediaPlayer does not support "
+ "setSurface() on this version of the platform.");
}
} else if (mSurfaceHolder != null) {
mMediaPlayer.setDisplay(mSurfaceHolder);
} else {
mMediaPlayer.setDisplay(null);
}
}
public static abstract class Callback {
public void onError() {}
public void onCompletion() {}
public void onStatusChanged() {}
public void onSizeChanged(int width, int height) {}
}
private static final class ICSMediaPlayer {
public static final void setSurface(MediaPlayer player, Surface surface) {
player.setSurface(surface);
}
}
}

View File

@@ -18,8 +18,10 @@ package com.example.android.supportv7.media;
import com.example.android.supportv7.R;
import android.content.Context;
import android.graphics.Point;
import android.graphics.SurfaceTexture;
import android.hardware.display.DisplayManager;
import android.os.Build;
import android.util.Log;
import android.view.Display;
import android.util.DisplayMetrics;
@@ -28,6 +30,8 @@ import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.ScaleGestureDetector;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.TextureView;
import android.view.View;
import android.view.Surface;
@@ -38,38 +42,147 @@ import android.widget.TextView;
/**
* Manages an overlay display window, used for simulating remote playback.
*/
public class OverlayDisplayWindow {
public abstract class OverlayDisplayWindow {
private static final String TAG = "OverlayDisplayWindow";
private static final boolean DEBUG = false;
private final float INITIAL_SCALE = 0.5f;
private final float MIN_SCALE = 0.3f;
private final float MAX_SCALE = 1.0f;
private final float WINDOW_ALPHA = 0.8f;
private static final float WINDOW_ALPHA = 0.8f;
private static final float INITIAL_SCALE = 0.5f;
private static final float MIN_SCALE = 0.3f;
private static final float MAX_SCALE = 1.0f;
protected final Context mContext;
protected final String mName;
protected final int mWidth;
protected final int mHeight;
protected final int mGravity;
protected OverlayWindowListener mListener;
protected OverlayDisplayWindow(Context context, String name,
int width, int height, int gravity) {
mContext = context;
mName = name;
mWidth = width;
mHeight = height;
mGravity = gravity;
}
public static OverlayDisplayWindow create(Context context, String name,
int width, int height, int gravity) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
return new JellybeanMr1Impl(context, name, width, height, gravity);
} else {
return new LegacyImpl(context, name, width, height, gravity);
}
}
public void setOverlayWindowListener(OverlayWindowListener listener) {
mListener = listener;
}
public Context getContext() {
return mContext;
}
public abstract void show();
public abstract void dismiss();
public abstract void updateAspectRatio(int width, int height);
// Watches for significant changes in the overlay display window lifecycle.
public interface OverlayWindowListener {
public void onWindowCreated(Surface surface);
public void onWindowCreated(SurfaceHolder surfaceHolder);
public void onWindowDestroyed();
}
/**
* Implementation for older versions.
*/
private static final class LegacyImpl extends OverlayDisplayWindow {
private final WindowManager mWindowManager;
private boolean mWindowVisible;
private SurfaceView mSurfaceView;
public LegacyImpl(Context context, String name,
int width, int height, int gravity) {
super(context, name, width, height, gravity);
mWindowManager = (WindowManager)context.getSystemService(
Context.WINDOW_SERVICE);
}
@Override
public void show() {
if (!mWindowVisible) {
mSurfaceView = new SurfaceView(mContext);
Display display = mWindowManager.getDefaultDisplay();
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
params.flags |= WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN
| WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE;
params.alpha = WINDOW_ALPHA;
params.gravity = Gravity.LEFT | Gravity.BOTTOM;
params.setTitle(mName);
int width = (int)(display.getWidth() * INITIAL_SCALE);
int height = (int)(display.getHeight() * INITIAL_SCALE);
if (mWidth > mHeight) {
height = mHeight * width / mWidth;
} else {
width = mWidth * height / mHeight;
}
params.width = width;
params.height = height;
mWindowManager.addView(mSurfaceView, params);
mWindowVisible = true;
mListener.onWindowCreated(mSurfaceView.getHolder());
}
}
@Override
public void dismiss() {
if (mWindowVisible) {
mListener.onWindowDestroyed();
mWindowManager.removeView(mSurfaceView);
mWindowVisible = false;
}
}
@Override
public void updateAspectRatio(int width, int height) {
}
}
/**
* Implementation for API version 17+.
*/
private static final class JellybeanMr1Impl extends OverlayDisplayWindow {
// When true, disables support for moving and resizing the overlay.
// The window is made non-touchable, which makes it possible to
// directly interact with the content underneath.
private final boolean DISABLE_MOVE_AND_RESIZE = false;
private final Context mContext;
private final int mWidth;
private final int mHeight;
private final int mGravity;
private OverlayWindowListener mListener;
private final String mTitle;
private static final boolean DISABLE_MOVE_AND_RESIZE = false;
private final DisplayManager mDisplayManager;
private final WindowManager mWindowManager;
private final Display mDefaultDisplay;
private final DisplayMetrics mDefaultDisplayMetrics = new DisplayMetrics();
private View mWindowContent;
private WindowManager.LayoutParams mWindowParams;
private TextureView mTextureView;
private TextView mTitleTextView;
private TextView mNameTextView;
private GestureDetector mGestureDetector;
private ScaleGestureDetector mScaleGestureDetector;
@@ -83,13 +196,9 @@ public class OverlayDisplayWindow {
private float mLiveTranslationY;
private float mLiveScale = 1.0f;
public OverlayDisplayWindow(Context context, String name,
public JellybeanMr1Impl(Context context, String name,
int width, int height, int gravity) {
mContext = context;
mWidth = width;
mHeight = height;
mGravity = gravity;
mTitle = name;
super(context, name, width, height, gravity);
mDisplayManager = (DisplayManager)context.getSystemService(
Context.DISPLAY_SERVICE);
@@ -102,14 +211,7 @@ public class OverlayDisplayWindow {
createWindow();
}
public void setOverlayWindowListener(OverlayWindowListener listener) {
mListener = listener;
}
public Context getContext() {
return mContext;
}
@Override
public void show() {
if (!mWindowVisible) {
mDisplayManager.registerDisplayListener(mDisplayListener, null);
@@ -125,6 +227,7 @@ public class OverlayDisplayWindow {
}
}
@Override
public void dismiss() {
if (mWindowVisible) {
mDisplayManager.unregisterDisplayListener(mDisplayListener);
@@ -133,13 +236,7 @@ public class OverlayDisplayWindow {
}
}
public void relayout() {
if (mWindowVisible) {
updateWindowParams();
mWindowManager.updateViewLayout(mWindowContent, mWindowParams);
}
}
@Override
public void updateAspectRatio(int width, int height) {
if (mWidth * height < mHeight * width) {
mTextureView.getLayoutParams().width = mWidth;
@@ -151,6 +248,13 @@ public class OverlayDisplayWindow {
relayout();
}
private void relayout() {
if (mWindowVisible) {
updateWindowParams();
mWindowManager.updateViewLayout(mWindowContent, mWindowParams);
}
}
private boolean updateDefaultDisplayInfo() {
mDefaultDisplay.getMetrics(mDefaultDisplayMetrics);
return true;
@@ -172,9 +276,9 @@ public class OverlayDisplayWindow {
mTextureView.setOpaque(false);
mTextureView.setSurfaceTextureListener(mSurfaceTextureListener);
mTitleTextView = (TextView)mWindowContent.findViewById(
mNameTextView = (TextView)mWindowContent.findViewById(
R.id.overlay_display_window_title);
mTitleTextView.setText(mTitle);
mNameTextView.setText(mName);
mWindowParams = new WindowManager.LayoutParams(
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
@@ -188,7 +292,7 @@ public class OverlayDisplayWindow {
}
mWindowParams.alpha = WINDOW_ALPHA;
mWindowParams.gravity = Gravity.TOP | Gravity.LEFT;
mWindowParams.setTitle(mTitle);
mWindowParams.setTitle(mName);
mGestureDetector = new GestureDetector(mContext, mOnGestureListener);
mScaleGestureDetector = new ScaleGestureDetector(mContext, mOnScaleGestureListener);
@@ -353,10 +457,5 @@ public class OverlayDisplayWindow {
return true;
}
};
// Watches for significant changes in the overlay display window lifecycle.
public interface OverlayWindowListener {
public void onWindowCreated(Surface surface);
public void onWindowDestroyed();
}
}

View File

@@ -35,8 +35,10 @@ import android.support.v7.media.MediaRouter.ControlRequestCallback;
import android.support.v7.media.MediaRouteProviderDescriptor;
import android.support.v7.media.MediaRouteDescriptor;
import android.util.Log;
import android.widget.Toast;
import android.view.Gravity;
import android.view.Surface;
import android.view.SurfaceHolder;
import java.util.ArrayList;
/**
@@ -188,26 +190,39 @@ final class SampleMediaRouteProvider extends MediaRouteProvider {
setDescriptor(providerDescriptor);
}
private void showToast(String msg) {
Toast toast = Toast.makeText(getContext(),
"[provider] " + msg, Toast.LENGTH_LONG);
toast.setGravity(Gravity.TOP, 0, 100);
toast.show();
}
private final class SampleRouteController extends MediaRouteProvider.RouteController {
private final String mRouteId;
// Create an overlay display window (used for simulating the remote playback only)
private final OverlayDisplayWindow mOverlay = new OverlayDisplayWindow(getContext(),
getContext().getResources().getString(R.string.sample_media_route_provider_remote),
1024, 768, Gravity.CENTER);
private final MediaPlayerWrapper mMediaPlayer = new MediaPlayerWrapper(getContext());
private final MediaSessionManager mSessionManager = new MediaSessionManager();
private final OverlayDisplayWindow mOverlay;
private final MediaPlayerWrapper mMediaPlayer;
private final MediaSessionManager mSessionManager;
public SampleRouteController(String routeId) {
mRouteId = routeId;
mMediaPlayer = new MediaPlayerWrapper(getContext());
mSessionManager = new MediaSessionManager();
mSessionManager.setCallback(mMediaPlayer);
mOverlay.setOverlayWindowListener(mMediaPlayer);
// Create an overlay display window (used for simulating the remote playback only)
mOverlay = OverlayDisplayWindow.create(getContext(),
getContext().getResources().getString(
R.string.sample_media_route_provider_remote),
1024, 768, Gravity.CENTER);
mOverlay.setOverlayWindowListener(new OverlayDisplayWindow.OverlayWindowListener() {
@Override
public void onWindowCreated(Surface surface) {
mMediaPlayer.setSurface(surface);
}
@Override
public void onWindowCreated(SurfaceHolder surfaceHolder) {
mMediaPlayer.setSurface(surfaceHolder);
}
@Override
public void onWindowDestroyed() {
}
});
mMediaPlayer.setCallback(new MediaPlayerCallback());
Log.d(TAG, mRouteId + ": Controller created");
}

View File

@@ -31,7 +31,6 @@ import android.media.AudioManager;
import android.media.AudioManager.OnAudioFocusChangeListener;
import android.media.MediaMetadataRetriever;
import android.media.RemoteControlClient;
import android.media.RemoteControlClient.MetadataEditor;
import android.net.Uri;
import android.os.Build;
import android.os.Handler;
@@ -58,14 +57,12 @@ import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.view.Display;
import android.view.Surface;
import android.view.SurfaceView;
import android.view.SurfaceHolder;
import android.view.WindowManager;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.ArrayAdapter;
import android.widget.Button;
import android.widget.ImageButton;
import android.widget.ListView;
import android.widget.TextView;
@@ -78,8 +75,6 @@ import android.widget.SeekBar;
import android.widget.SeekBar.OnSeekBarChangeListener;
import java.util.ArrayList;
import java.util.List;
import java.io.File;
/**
@@ -898,7 +893,7 @@ public class SampleMediaRouterActivity extends ActionBarActivity {
public void onFinish(boolean error) {
MediaQueueItem item = mSessionManager.finish(error);
updateUi();
if (error) {
if (error && item != null) {
showToast("Failed to play item " + item.getUri());
}
}
@@ -908,13 +903,13 @@ public class SampleMediaRouterActivity extends ActionBarActivity {
public void surfaceChanged(SurfaceHolder holder, int format,
int width, int height) {
Log.d(TAG, "surfaceChanged "+width+"x"+height);
mMediaPlayer.onWindowCreated(holder.getSurface());
mMediaPlayer.setSurface(holder);
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
Log.d(TAG, "surfaceCreated");
mMediaPlayer.onWindowCreated(holder.getSurface());
mMediaPlayer.setSurface(holder);
mLocalPlayer.updateSize(mVideoWidth, mVideoHeight);
}
@@ -1007,7 +1002,7 @@ public class SampleMediaRouterActivity extends ActionBarActivity {
};
private final class DemoPresentation extends Presentation {
private SurfaceView mSurfaceView;
private SurfaceView mPresentationSurfaceView;
public DemoPresentation(Context context, Display display) {
super(context, display);
@@ -1026,16 +1021,16 @@ public class SampleMediaRouterActivity extends ActionBarActivity {
// Inflate the layout.
setContentView(R.layout.sample_media_router_presentation);
// Set up the surface view for visual interest.
mSurfaceView = (SurfaceView)findViewById(R.id.surface_view);
SurfaceHolder holder = mSurfaceView.getHolder();
// Set up the surface view.
mPresentationSurfaceView = (SurfaceView)findViewById(R.id.surface_view);
SurfaceHolder holder = mPresentationSurfaceView.getHolder();
holder.addCallback(mLocalPlayer);
}
public void updateSize(int width, int height) {
int surfaceHeight = getWindow().getDecorView().getHeight();
int surfaceWidth = getWindow().getDecorView().getWidth();
ViewGroup.LayoutParams lp = mSurfaceView.getLayoutParams();
ViewGroup.LayoutParams lp = mPresentationSurfaceView.getLayoutParams();
if (surfaceWidth * height < surfaceHeight * width) {
lp.width = surfaceWidth;
lp.height = surfaceWidth * height / width;
@@ -1044,11 +1039,7 @@ public class SampleMediaRouterActivity extends ActionBarActivity {
lp.height = surfaceHeight;
}
Log.d(TAG, "video rect is " + lp.width + "x" + lp.height);
mSurfaceView.setLayoutParams(lp);
}
public void clearContent() {
//TO-DO: clear surface view
mPresentationSurfaceView.setLayoutParams(lp);
}
}
}