Add AnimatedVectorDrawable support for VideoPreference
- We are planning to use animation vector drawable to replace mp4 file to reduce apk size - Add vectorAnimation attr in VideoPreference - Delegate VideoPreference media control to AnimationController Bug: 143270527 Test: manual, robolectric Change-Id: Ia5859f928a9082085cdf715c762f964e1c99e003
This commit is contained in:
@@ -119,6 +119,7 @@
|
|||||||
<declare-styleable name="VideoPreference">
|
<declare-styleable name="VideoPreference">
|
||||||
<attr name="animation" format="reference" />
|
<attr name="animation" format="reference" />
|
||||||
<attr name="preview" format="reference" />
|
<attr name="preview" format="reference" />
|
||||||
|
<attr name="vectorAnimation" format="reference" />
|
||||||
</declare-styleable>
|
</declare-styleable>
|
||||||
|
|
||||||
<!-- For AspectRatioFrameLayout -->
|
<!-- For AspectRatioFrameLayout -->
|
||||||
|
149
src/com/android/settings/widget/MediaAnimationController.java
Normal file
149
src/com/android/settings/widget/MediaAnimationController.java
Normal file
@@ -0,0 +1,149 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.widget;
|
||||||
|
|
||||||
|
import android.content.ContentResolver;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.SurfaceTexture;
|
||||||
|
import android.media.MediaPlayer;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.view.Surface;
|
||||||
|
import android.view.TextureView;
|
||||||
|
import android.view.View;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link VideoPreference.AnimationController} containing a {@link
|
||||||
|
* MediaPlayer}. The controller is used by {@link VideoPreference} to display
|
||||||
|
* a mp4 resource.
|
||||||
|
*/
|
||||||
|
class MediaAnimationController implements VideoPreference.AnimationController {
|
||||||
|
private MediaPlayer mMediaPlayer;
|
||||||
|
private boolean mVideoReady;
|
||||||
|
|
||||||
|
MediaAnimationController(Context context, int videoId) {
|
||||||
|
final Uri videoPath = new Uri.Builder().scheme(ContentResolver.SCHEME_ANDROID_RESOURCE)
|
||||||
|
.authority(context.getPackageName())
|
||||||
|
.appendPath(String.valueOf(videoId))
|
||||||
|
.build();
|
||||||
|
mMediaPlayer = MediaPlayer.create(context, videoPath);
|
||||||
|
// when the playback res is invalid or others, MediaPlayer create may fail
|
||||||
|
// and return null, so need add the null judgement.
|
||||||
|
if (mMediaPlayer != null) {
|
||||||
|
mMediaPlayer.seekTo(0);
|
||||||
|
mMediaPlayer.setOnSeekCompleteListener(mp -> mVideoReady = true);
|
||||||
|
mMediaPlayer.setOnPreparedListener(mediaPlayer -> mediaPlayer.setLooping(true));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getVideoWidth() {
|
||||||
|
return mMediaPlayer.getVideoWidth();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getVideoHeight() {
|
||||||
|
return mMediaPlayer.getVideoHeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void pause() {
|
||||||
|
mMediaPlayer.pause();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start() {
|
||||||
|
mMediaPlayer.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isPlaying() {
|
||||||
|
return mMediaPlayer.isPlaying();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getDuration() {
|
||||||
|
return mMediaPlayer.getDuration();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void attachView(TextureView video, View preview, View playButton) {
|
||||||
|
updateViewStates(preview, playButton);
|
||||||
|
video.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() {
|
||||||
|
@Override
|
||||||
|
public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width,
|
||||||
|
int height) {
|
||||||
|
if (mMediaPlayer != null) {
|
||||||
|
final Surface surface = new Surface(surfaceTexture);
|
||||||
|
mMediaPlayer.setSurface(surface);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSurfaceTextureSizeChanged(SurfaceTexture surfaceTexture, int width,
|
||||||
|
int height) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onSurfaceTextureDestroyed(SurfaceTexture surfaceTexture) {
|
||||||
|
preview.setVisibility(View.VISIBLE);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) {
|
||||||
|
if (mVideoReady) {
|
||||||
|
if (preview.getVisibility() == View.VISIBLE) {
|
||||||
|
preview.setVisibility(View.GONE);
|
||||||
|
}
|
||||||
|
if (mMediaPlayer != null
|
||||||
|
&& !mMediaPlayer.isPlaying()) {
|
||||||
|
mMediaPlayer.start();
|
||||||
|
playButton.setVisibility(View.GONE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (mMediaPlayer != null && !mMediaPlayer.isPlaying()
|
||||||
|
&& playButton.getVisibility() != View.VISIBLE) {
|
||||||
|
playButton.setVisibility(View.VISIBLE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
video.setOnClickListener(v -> updateViewStates(preview, playButton));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void release() {
|
||||||
|
if (mMediaPlayer != null) {
|
||||||
|
mMediaPlayer.stop();
|
||||||
|
mMediaPlayer.reset();
|
||||||
|
mMediaPlayer.release();
|
||||||
|
mMediaPlayer = null;
|
||||||
|
mVideoReady = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateViewStates(View imageView, View playButton) {
|
||||||
|
if (mMediaPlayer.isPlaying()) {
|
||||||
|
mMediaPlayer.pause();
|
||||||
|
playButton.setVisibility(View.VISIBLE);
|
||||||
|
imageView.setVisibility(View.VISIBLE);
|
||||||
|
} else {
|
||||||
|
imageView.setVisibility(View.GONE);
|
||||||
|
playButton.setVisibility(View.GONE);
|
||||||
|
mMediaPlayer.start();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
111
src/com/android/settings/widget/VectorAnimationController.java
Normal file
111
src/com/android/settings/widget/VectorAnimationController.java
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.widget;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.drawable.Drawable;
|
||||||
|
import android.view.TextureView;
|
||||||
|
import android.view.View;
|
||||||
|
|
||||||
|
import androidx.vectordrawable.graphics.drawable.Animatable2Compat;
|
||||||
|
import androidx.vectordrawable.graphics.drawable.AnimatedVectorDrawableCompat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A {@link VideoPreference.AnimationController} containing a {@link
|
||||||
|
* AnimatedVectorDrawableCompat}. The controller is used by {@link VideoPreference}
|
||||||
|
* to display AnimatedVectorDrawable content.
|
||||||
|
*/
|
||||||
|
class VectorAnimationController implements VideoPreference.AnimationController {
|
||||||
|
private AnimatedVectorDrawableCompat mAnimatedVectorDrawableCompat;
|
||||||
|
private Drawable mPreviewDrawable;
|
||||||
|
private Animatable2Compat.AnimationCallback mAnimationCallback;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called by a preference panel fragment to finish itself.
|
||||||
|
*
|
||||||
|
* @param context Application Context
|
||||||
|
* @param animationId An {@link android.graphics.drawable.AnimationDrawable} resource id
|
||||||
|
*/
|
||||||
|
VectorAnimationController(Context context, int animationId) {
|
||||||
|
mAnimatedVectorDrawableCompat = AnimatedVectorDrawableCompat.create(context, animationId);
|
||||||
|
mAnimationCallback = new Animatable2Compat.AnimationCallback() {
|
||||||
|
@Override
|
||||||
|
public void onAnimationEnd(Drawable drawable) {
|
||||||
|
mAnimatedVectorDrawableCompat.start();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getVideoWidth() {
|
||||||
|
return mAnimatedVectorDrawableCompat.getIntrinsicWidth();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getVideoHeight() {
|
||||||
|
return mAnimatedVectorDrawableCompat.getIntrinsicHeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void pause() {
|
||||||
|
mAnimatedVectorDrawableCompat.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void start() {
|
||||||
|
mAnimatedVectorDrawableCompat.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isPlaying() {
|
||||||
|
return mAnimatedVectorDrawableCompat.isRunning();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getDuration() {
|
||||||
|
// We can't get duration from AnimatedVectorDrawable, just return a non zero value.
|
||||||
|
return 5000;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void attachView(TextureView video, View preview, View playButton) {
|
||||||
|
mPreviewDrawable = preview.getForeground();
|
||||||
|
video.setVisibility(View.GONE);
|
||||||
|
updateViewStates(preview, playButton);
|
||||||
|
preview.setOnClickListener(v -> updateViewStates(preview, playButton));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void release() {
|
||||||
|
mAnimatedVectorDrawableCompat.stop();
|
||||||
|
mAnimatedVectorDrawableCompat.clearAnimationCallbacks();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateViewStates(View imageView, View playButton) {
|
||||||
|
if (mAnimatedVectorDrawableCompat.isRunning()) {
|
||||||
|
mAnimatedVectorDrawableCompat.stop();
|
||||||
|
mAnimatedVectorDrawableCompat.clearAnimationCallbacks();
|
||||||
|
playButton.setVisibility(View.VISIBLE);
|
||||||
|
imageView.setForeground(mPreviewDrawable);
|
||||||
|
} else {
|
||||||
|
playButton.setVisibility(View.GONE);
|
||||||
|
imageView.setForeground(mAnimatedVectorDrawableCompat);
|
||||||
|
mAnimatedVectorDrawableCompat.start();
|
||||||
|
mAnimatedVectorDrawableCompat.registerAnimationCallback(mAnimationCallback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -16,16 +16,11 @@
|
|||||||
|
|
||||||
package com.android.settings.widget;
|
package com.android.settings.widget;
|
||||||
|
|
||||||
import android.content.ContentResolver;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.res.TypedArray;
|
import android.content.res.TypedArray;
|
||||||
import android.graphics.SurfaceTexture;
|
|
||||||
import android.media.MediaPlayer;
|
|
||||||
import android.net.Uri;
|
|
||||||
import android.util.AttributeSet;
|
import android.util.AttributeSet;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.util.TypedValue;
|
import android.util.TypedValue;
|
||||||
import android.view.Surface;
|
|
||||||
import android.view.TextureView;
|
import android.view.TextureView;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.ImageView;
|
import android.widget.ImageView;
|
||||||
@@ -34,30 +29,27 @@ import android.widget.LinearLayout;
|
|||||||
import androidx.annotation.VisibleForTesting;
|
import androidx.annotation.VisibleForTesting;
|
||||||
import androidx.preference.Preference;
|
import androidx.preference.Preference;
|
||||||
import androidx.preference.PreferenceViewHolder;
|
import androidx.preference.PreferenceViewHolder;
|
||||||
|
import androidx.vectordrawable.graphics.drawable.AnimatedVectorDrawableCompat;
|
||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A full width preference that hosts a MP4 video.
|
* A full width preference that hosts a MP4 video or a {@link AnimatedVectorDrawableCompat}.
|
||||||
*/
|
*/
|
||||||
public class VideoPreference extends Preference {
|
public class VideoPreference extends Preference {
|
||||||
|
|
||||||
private static final String TAG = "VideoPreference";
|
private static final String TAG = "VideoPreference";
|
||||||
private final Context mContext;
|
private final Context mContext;
|
||||||
|
|
||||||
private Uri mVideoPath;
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
MediaPlayer mMediaPlayer;
|
AnimationController mAnimationController;
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
boolean mAnimationAvailable;
|
boolean mAnimationAvailable;
|
||||||
@VisibleForTesting
|
|
||||||
boolean mVideoReady;
|
|
||||||
private boolean mVideoPaused;
|
private boolean mVideoPaused;
|
||||||
private float mAspectRatio = 1.0f;
|
private float mAspectRatio = 1.0f;
|
||||||
private int mPreviewResource;
|
private int mPreviewId;
|
||||||
private boolean mViewVisible;
|
|
||||||
private Surface mSurface;
|
|
||||||
private int mAnimationId;
|
private int mAnimationId;
|
||||||
|
private int mVectorAnimationId;
|
||||||
private int mHeight = LinearLayout.LayoutParams.MATCH_PARENT - 1; // video height in pixels
|
private int mHeight = LinearLayout.LayoutParams.MATCH_PARENT - 1; // video height in pixels
|
||||||
|
|
||||||
public VideoPreference(Context context) {
|
public VideoPreference(Context context) {
|
||||||
@@ -84,19 +76,17 @@ public class VideoPreference extends Preference {
|
|||||||
mAnimationId = mAnimationId == 0
|
mAnimationId = mAnimationId == 0
|
||||||
? attributes.getResourceId(R.styleable.VideoPreference_animation, 0)
|
? attributes.getResourceId(R.styleable.VideoPreference_animation, 0)
|
||||||
: mAnimationId;
|
: mAnimationId;
|
||||||
mVideoPath = new Uri.Builder().scheme(ContentResolver.SCHEME_ANDROID_RESOURCE)
|
mPreviewId = mPreviewId == 0
|
||||||
.authority(context.getPackageName())
|
? attributes.getResourceId(R.styleable.VideoPreference_preview, 0)
|
||||||
.appendPath(String.valueOf(mAnimationId))
|
: mPreviewId;
|
||||||
.build();
|
mVectorAnimationId = attributes.getResourceId(
|
||||||
mPreviewResource = mPreviewResource == 0
|
R.styleable.VideoPreference_vectorAnimation, 0);
|
||||||
? attributes.getResourceId(R.styleable.VideoPreference_preview, 0)
|
if (mPreviewId == 0 && mAnimationId == 0 && mVectorAnimationId == 0) {
|
||||||
: mPreviewResource;
|
|
||||||
if (mPreviewResource == 0 && mAnimationId == 0) {
|
|
||||||
setVisible(false);
|
setVisible(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
initMediaPlayer();
|
initAnimationController();
|
||||||
if (mMediaPlayer != null && mMediaPlayer.getDuration() > 0) {
|
if (mAnimationController != null && mAnimationController.getDuration() > 0) {
|
||||||
setVisible(true);
|
setVisible(true);
|
||||||
setLayoutResource(R.layout.video_preference);
|
setLayoutResource(R.layout.video_preference);
|
||||||
mAnimationAvailable = true;
|
mAnimationAvailable = true;
|
||||||
@@ -120,135 +110,63 @@ public class VideoPreference extends Preference {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final TextureView video = (TextureView) holder.findViewById(R.id.video_texture_view);
|
final TextureView video = (TextureView) holder.findViewById(R.id.video_texture_view);
|
||||||
final ImageView imageView = (ImageView) holder.findViewById(R.id.video_preview_image);
|
final ImageView previewImage = (ImageView) holder.findViewById(R.id.video_preview_image);
|
||||||
final ImageView playButton = (ImageView) holder.findViewById(R.id.video_play_button);
|
final ImageView playButton = (ImageView) holder.findViewById(R.id.video_play_button);
|
||||||
final AspectRatioFrameLayout layout = (AspectRatioFrameLayout) holder.findViewById(
|
final AspectRatioFrameLayout layout = (AspectRatioFrameLayout) holder.findViewById(
|
||||||
R.id.video_container);
|
R.id.video_container);
|
||||||
|
|
||||||
imageView.setImageResource(mPreviewResource);
|
previewImage.setImageResource(mPreviewId);
|
||||||
layout.setAspectRatio(mAspectRatio);
|
layout.setAspectRatio(mAspectRatio);
|
||||||
if (mHeight >= LinearLayout.LayoutParams.MATCH_PARENT) {
|
if (mHeight >= LinearLayout.LayoutParams.MATCH_PARENT) {
|
||||||
layout.setLayoutParams(new LinearLayout.LayoutParams(
|
layout.setLayoutParams(new LinearLayout.LayoutParams(
|
||||||
LinearLayout.LayoutParams.MATCH_PARENT, mHeight));
|
LinearLayout.LayoutParams.MATCH_PARENT, mHeight));
|
||||||
}
|
}
|
||||||
updateViewStates(imageView, playButton);
|
mAnimationController.attachView(video, previewImage, playButton);
|
||||||
|
|
||||||
video.setOnClickListener(v -> updateViewStates(imageView, playButton));
|
|
||||||
|
|
||||||
video.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() {
|
|
||||||
@Override
|
|
||||||
public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width,
|
|
||||||
int height) {
|
|
||||||
if (mMediaPlayer != null) {
|
|
||||||
mSurface = new Surface(surfaceTexture);
|
|
||||||
mMediaPlayer.setSurface(mSurface);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onSurfaceTextureSizeChanged(SurfaceTexture surfaceTexture, int width,
|
|
||||||
int height) {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onSurfaceTextureDestroyed(SurfaceTexture surfaceTexture) {
|
|
||||||
imageView.setVisibility(View.VISIBLE);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) {
|
|
||||||
if (!mViewVisible) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (mVideoReady) {
|
|
||||||
if (imageView.getVisibility() == View.VISIBLE) {
|
|
||||||
imageView.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
if (!mVideoPaused && mMediaPlayer != null && !mMediaPlayer.isPlaying()) {
|
|
||||||
mMediaPlayer.start();
|
|
||||||
playButton.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (mMediaPlayer != null && !mMediaPlayer.isPlaying() &&
|
|
||||||
playButton.getVisibility() != View.VISIBLE) {
|
|
||||||
playButton.setVisibility(View.VISIBLE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@VisibleForTesting
|
|
||||||
void updateViewStates(ImageView imageView, ImageView playButton) {
|
|
||||||
if (mMediaPlayer != null) {
|
|
||||||
if (mMediaPlayer.isPlaying()) {
|
|
||||||
mMediaPlayer.pause();
|
|
||||||
playButton.setVisibility(View.VISIBLE);
|
|
||||||
imageView.setVisibility(View.VISIBLE);
|
|
||||||
mVideoPaused = true;
|
|
||||||
} else {
|
|
||||||
imageView.setVisibility(View.GONE);
|
|
||||||
playButton.setVisibility(View.GONE);
|
|
||||||
mMediaPlayer.start();
|
|
||||||
mVideoPaused = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onDetached() {
|
public void onDetached() {
|
||||||
releaseMediaPlayer();
|
releaseAnimationController();
|
||||||
super.onDetached();
|
super.onDetached();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void onViewVisible(boolean videoPaused) {
|
public void onViewVisible(boolean videoPaused) {
|
||||||
mViewVisible = true;
|
|
||||||
mVideoPaused = videoPaused;
|
mVideoPaused = videoPaused;
|
||||||
initMediaPlayer();
|
initAnimationController();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void onViewInvisible() {
|
public void onViewInvisible() {
|
||||||
mViewVisible = false;
|
releaseAnimationController();
|
||||||
releaseMediaPlayer();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the video for this preference. If a previous video was set this one will override it
|
* Sets the video for this preference. If a previous video was set this one will override it
|
||||||
* and properly release any resources and re-initialize the preference to play the new video.
|
* and properly release any resources and re-initialize the preference to play the new video.
|
||||||
*
|
*
|
||||||
* @param videoId The raw res id of the video
|
* @param videoId The raw res id of the video
|
||||||
* @param previewId The drawable res id of the preview image to use if the video fails to load.
|
* @param previewId The drawable res id of the preview image to use if the video fails to load.
|
||||||
*/
|
*/
|
||||||
public void setVideo(int videoId, int previewId) {
|
public void setVideo(int videoId, int previewId) {
|
||||||
mAnimationId = videoId;
|
mAnimationId = videoId;
|
||||||
mPreviewResource = previewId;
|
mPreviewId = previewId;
|
||||||
releaseMediaPlayer();
|
releaseAnimationController();
|
||||||
initialize(mContext, null);
|
initialize(mContext, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void initMediaPlayer() {
|
private void initAnimationController() {
|
||||||
if (mMediaPlayer == null) {
|
if (mVectorAnimationId != 0) {
|
||||||
mMediaPlayer = MediaPlayer.create(mContext, mVideoPath);
|
mAnimationController = new VectorAnimationController(mContext, mVectorAnimationId);
|
||||||
// when the playback res is invalid or others, MediaPlayer create may fail
|
return;
|
||||||
// and return null, so need add the null judgement.
|
}
|
||||||
if (mMediaPlayer != null) {
|
if (mAnimationId != 0) {
|
||||||
mMediaPlayer.seekTo(0);
|
mAnimationController = new MediaAnimationController(mContext, mAnimationId);
|
||||||
mMediaPlayer.setOnSeekCompleteListener(mp -> mVideoReady = true);
|
|
||||||
mMediaPlayer.setOnPreparedListener(mediaPlayer -> mediaPlayer.setLooping(true));
|
|
||||||
if (mSurface != null) {
|
|
||||||
mMediaPlayer.setSurface(mSurface);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void releaseMediaPlayer() {
|
private void releaseAnimationController() {
|
||||||
if (mMediaPlayer != null) {
|
if (mAnimationController != null) {
|
||||||
mMediaPlayer.stop();
|
mAnimationController.release();
|
||||||
mMediaPlayer.reset();
|
mAnimationController = null;
|
||||||
mMediaPlayer.release();
|
|
||||||
mMediaPlayer = null;
|
|
||||||
mVideoReady = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -262,6 +180,7 @@ public class VideoPreference extends Preference {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* sets the height of the video preference
|
* sets the height of the video preference
|
||||||
|
*
|
||||||
* @param height in dp
|
* @param height in dp
|
||||||
*/
|
*/
|
||||||
public void setHeight(float height) {
|
public void setHeight(float height) {
|
||||||
@@ -271,6 +190,52 @@ public class VideoPreference extends Preference {
|
|||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
void updateAspectRatio() {
|
void updateAspectRatio() {
|
||||||
mAspectRatio = mMediaPlayer.getVideoWidth() / (float) mMediaPlayer.getVideoHeight();
|
mAspectRatio = mAnimationController.getVideoWidth()
|
||||||
|
/ (float) mAnimationController.getVideoHeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handle animation operations.
|
||||||
|
*/
|
||||||
|
interface AnimationController {
|
||||||
|
/**
|
||||||
|
* Pauses the animation.
|
||||||
|
*/
|
||||||
|
void pause();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts the animation.
|
||||||
|
*/
|
||||||
|
void start();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Releases the animation object.
|
||||||
|
*/
|
||||||
|
void release();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attaches the animation to UI view.
|
||||||
|
*/
|
||||||
|
void attachView(TextureView video, View preview, View playButton);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the animation Width.
|
||||||
|
*/
|
||||||
|
int getVideoWidth();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the animation Height.
|
||||||
|
*/
|
||||||
|
int getVideoHeight();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the animation duration.
|
||||||
|
*/
|
||||||
|
int getDuration();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns if the animation is playing.
|
||||||
|
*/
|
||||||
|
boolean isPlaying();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,51 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.testutils.shadow;
|
||||||
|
|
||||||
|
import static org.robolectric.shadows.ShadowMediaPlayer.State.INITIALIZED;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.media.MediaPlayer;
|
||||||
|
import android.net.Uri;
|
||||||
|
|
||||||
|
import org.robolectric.annotation.Implementation;
|
||||||
|
import org.robolectric.annotation.Implements;
|
||||||
|
import org.robolectric.shadow.api.Shadow;
|
||||||
|
import org.robolectric.shadows.ShadowMediaPlayer;
|
||||||
|
import org.robolectric.shadows.util.DataSource;
|
||||||
|
|
||||||
|
@Implements(MediaPlayer.class)
|
||||||
|
public class ShadowSettingsMediaPlayer extends ShadowMediaPlayer {
|
||||||
|
|
||||||
|
@Implementation
|
||||||
|
public static MediaPlayer create(Context context, Uri uri) {
|
||||||
|
final DataSource ds = DataSource.toDataSource(context, uri);
|
||||||
|
addMediaInfo(ds, new ShadowMediaPlayer.MediaInfo());
|
||||||
|
|
||||||
|
final MediaPlayer mp = new MediaPlayer();
|
||||||
|
final ShadowMediaPlayer shadow = Shadow.extract(mp);
|
||||||
|
try {
|
||||||
|
shadow.setDataSource(ds);
|
||||||
|
shadow.setState(INITIALIZED);
|
||||||
|
mp.prepare();
|
||||||
|
} catch (Exception e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return mp;
|
||||||
|
}
|
||||||
|
}
|
@@ -18,26 +18,24 @@ package com.android.settings.widget;
|
|||||||
|
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
|
||||||
import static org.mockito.ArgumentMatchers.anyInt;
|
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
import static org.mockito.ArgumentMatchers.eq;
|
||||||
import static org.mockito.Mockito.doReturn;
|
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.never;
|
import static org.mockito.Mockito.never;
|
||||||
|
import static org.mockito.Mockito.spy;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.graphics.SurfaceTexture;
|
import android.graphics.SurfaceTexture;
|
||||||
import android.media.MediaPlayer;
|
|
||||||
import android.view.LayoutInflater;
|
import android.view.LayoutInflater;
|
||||||
import android.view.TextureView;
|
import android.view.TextureView;
|
||||||
|
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.ImageView;
|
import android.widget.ImageView;
|
||||||
|
|
||||||
import androidx.preference.PreferenceViewHolder;
|
import androidx.preference.PreferenceViewHolder;
|
||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.testutils.shadow.ShadowSettingsMediaPlayer;
|
||||||
|
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
@@ -46,14 +44,15 @@ import org.mockito.Mock;
|
|||||||
import org.mockito.MockitoAnnotations;
|
import org.mockito.MockitoAnnotations;
|
||||||
import org.robolectric.RobolectricTestRunner;
|
import org.robolectric.RobolectricTestRunner;
|
||||||
import org.robolectric.RuntimeEnvironment;
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
import org.robolectric.annotation.Config;
|
||||||
|
|
||||||
@RunWith(RobolectricTestRunner.class)
|
@RunWith(RobolectricTestRunner.class)
|
||||||
|
@Config(shadows = ShadowSettingsMediaPlayer.class)
|
||||||
public class VideoPreferenceTest {
|
public class VideoPreferenceTest {
|
||||||
private static final int VIDEO_WIDTH = 100;
|
private static final int VIDEO_WIDTH = 100;
|
||||||
private static final int VIDEO_HEIGHT = 150;
|
private static final int VIDEO_HEIGHT = 150;
|
||||||
|
|
||||||
@Mock
|
private VideoPreference.AnimationController mAnimationController;
|
||||||
private MediaPlayer mMediaPlayer;
|
|
||||||
@Mock
|
@Mock
|
||||||
private ImageView fakePreview;
|
private ImageView fakePreview;
|
||||||
@Mock
|
@Mock
|
||||||
@@ -67,10 +66,12 @@ public class VideoPreferenceTest {
|
|||||||
MockitoAnnotations.initMocks(this);
|
MockitoAnnotations.initMocks(this);
|
||||||
|
|
||||||
mContext = RuntimeEnvironment.application;
|
mContext = RuntimeEnvironment.application;
|
||||||
|
mAnimationController = spy(
|
||||||
|
new MediaAnimationController(mContext, R.raw.accessibility_screen_magnification));
|
||||||
mVideoPreference = new VideoPreference(mContext, null /* attrs */);
|
mVideoPreference = new VideoPreference(mContext, null /* attrs */);
|
||||||
mVideoPreference.mMediaPlayer = mMediaPlayer;
|
mVideoPreference.mAnimationController = mAnimationController;
|
||||||
when(mMediaPlayer.getVideoWidth()).thenReturn(VIDEO_WIDTH);
|
when(mAnimationController.getVideoWidth()).thenReturn(VIDEO_WIDTH);
|
||||||
when(mMediaPlayer.getVideoHeight()).thenReturn(VIDEO_HEIGHT);
|
when(mAnimationController.getVideoHeight()).thenReturn(VIDEO_HEIGHT);
|
||||||
|
|
||||||
mPreferenceViewHolder = PreferenceViewHolder.createInstanceForTests(
|
mPreferenceViewHolder = PreferenceViewHolder.createInstanceForTests(
|
||||||
LayoutInflater.from(mContext).inflate(R.layout.video_preference, null));
|
LayoutInflater.from(mContext).inflate(R.layout.video_preference, null));
|
||||||
@@ -91,17 +92,17 @@ public class VideoPreferenceTest {
|
|||||||
@Test
|
@Test
|
||||||
public void onSurfaceTextureUpdated_viewInvisible_shouldNotStartPlayingVideo() {
|
public void onSurfaceTextureUpdated_viewInvisible_shouldNotStartPlayingVideo() {
|
||||||
final TextureView video =
|
final TextureView video =
|
||||||
(TextureView) mPreferenceViewHolder.findViewById(R.id.video_texture_view);
|
(TextureView) mPreferenceViewHolder.findViewById(R.id.video_texture_view);
|
||||||
mVideoPreference.mAnimationAvailable = true;
|
mVideoPreference.mAnimationAvailable = true;
|
||||||
mVideoPreference.mVideoReady = true;
|
|
||||||
mVideoPreference.onViewInvisible();
|
|
||||||
mVideoPreference.onBindViewHolder(mPreferenceViewHolder);
|
mVideoPreference.onBindViewHolder(mPreferenceViewHolder);
|
||||||
when(mMediaPlayer.isPlaying()).thenReturn(false);
|
mAnimationController.attachView(video, fakePreview, fakePlayButton);
|
||||||
|
when(mAnimationController.isPlaying()).thenReturn(false);
|
||||||
final TextureView.SurfaceTextureListener listener = video.getSurfaceTextureListener();
|
final TextureView.SurfaceTextureListener listener = video.getSurfaceTextureListener();
|
||||||
|
|
||||||
|
mVideoPreference.onViewInvisible();
|
||||||
listener.onSurfaceTextureUpdated(mock(SurfaceTexture.class));
|
listener.onSurfaceTextureUpdated(mock(SurfaceTexture.class));
|
||||||
|
|
||||||
verify(mMediaPlayer, never()).start();
|
verify(mAnimationController, never()).start();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@@ -110,32 +111,30 @@ public class VideoPreferenceTest {
|
|||||||
|
|
||||||
mVideoPreference.onViewInvisible();
|
mVideoPreference.onViewInvisible();
|
||||||
|
|
||||||
verify(mMediaPlayer).release();
|
verify(mAnimationController).release();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void updateViewStates_paused_updatesViews() {
|
public void updateViewStates_paused_updatesViews() {
|
||||||
when(mMediaPlayer.isPlaying()).thenReturn(true);
|
mAnimationController.start();
|
||||||
mVideoPreference.updateViewStates(fakePreview, fakePlayButton);
|
|
||||||
|
mVideoPreference.mAnimationController.attachView(new TextureView(mContext), fakePreview,
|
||||||
|
fakePlayButton);
|
||||||
|
|
||||||
verify(fakePlayButton).setVisibility(eq(View.VISIBLE));
|
verify(fakePlayButton).setVisibility(eq(View.VISIBLE));
|
||||||
verify(fakePreview).setVisibility(eq(View.VISIBLE));
|
verify(fakePreview).setVisibility(eq(View.VISIBLE));
|
||||||
verify(mMediaPlayer).pause();
|
assertThat(mAnimationController.isPlaying()).isFalse();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void updateViewStates_playing_updatesViews() {
|
public void updateViewStates_playing_updatesViews() {
|
||||||
when(mMediaPlayer.isPlaying()).thenReturn(false);
|
mAnimationController.pause();
|
||||||
mVideoPreference.updateViewStates(fakePreview, fakePlayButton);
|
|
||||||
|
mVideoPreference.mAnimationController.attachView(new TextureView(mContext), fakePreview,
|
||||||
|
fakePlayButton);
|
||||||
|
|
||||||
verify(fakePlayButton).setVisibility(eq(View.GONE));
|
verify(fakePlayButton).setVisibility(eq(View.GONE));
|
||||||
verify(fakePreview).setVisibility(eq(View.GONE));
|
verify(fakePreview).setVisibility(eq(View.GONE));
|
||||||
verify(mMediaPlayer).start();
|
assertThat(mAnimationController.isPlaying()).isTrue();
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void updateViewStates_noMediaPlayer_skips() {
|
|
||||||
mVideoPreference.mMediaPlayer = null;
|
|
||||||
mVideoPreference.updateViewStates(fakePreview, fakePlayButton);
|
|
||||||
verify(fakePlayButton, never()).setVisibility(anyInt());
|
|
||||||
verify(fakePreview, never()).setVisibility(anyInt());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user