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:
Raff Tsai
2019-10-31 14:03:34 +08:00
parent e49ff7de26
commit 953da2ee40
6 changed files with 422 additions and 146 deletions

View File

@@ -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 -->

View 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();
}
}
}

View 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);
}
}
}

View File

@@ -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();
} }
} }

View File

@@ -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;
}
}

View File

@@ -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());
} }
} }