screen off. Re-initialize the animation video if necessary when the preference is visible to the user. Fixes: 31023261 Test: manual - 1. Go to Settings -> Moves and verify that the animation video are displayed correctly on load. 2. Press power button to turn screen off, then on again. Verify when Moves settings resumes, the animation video is displayed correctly. 3. Repeat #2 above but let display time out instead of turning off screen by power button. Change-Id: I0ee21d23a4c402b3a8463dca3b36fce67d651993
242 lines
8.3 KiB
Java
242 lines
8.3 KiB
Java
/*
|
|
* Copyright (C) 2016 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.gestures;
|
|
|
|
import android.app.LoaderManager;
|
|
import android.content.ContentResolver;
|
|
import android.content.Context;
|
|
import android.content.Loader;
|
|
import android.content.res.TypedArray;
|
|
import android.graphics.Bitmap;
|
|
import android.graphics.drawable.BitmapDrawable;
|
|
import android.graphics.SurfaceTexture;
|
|
import android.media.MediaMetadataRetriever;
|
|
import android.media.MediaPlayer;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.support.v14.preference.SwitchPreference;
|
|
import android.support.v7.preference.PreferenceViewHolder;
|
|
import android.view.View;
|
|
import android.view.Surface;
|
|
import android.view.TextureView;
|
|
import android.widget.ImageView;
|
|
import android.util.AttributeSet;
|
|
import android.util.Log;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.utils.AsyncLoader;
|
|
|
|
/**
|
|
* Preference item for a gesture with a switch to signify if it should be enabled.
|
|
* This shows the title and description of the gesture along with an animation showing how to do
|
|
* the gesture
|
|
*/
|
|
public final class GesturePreference extends SwitchPreference implements
|
|
LoaderManager.LoaderCallbacks<Bitmap> {
|
|
private static final String TAG = "GesturePreference";
|
|
private final Context mContext;
|
|
|
|
private Uri mVideoPath;
|
|
private MediaPlayer mMediaPlayer;
|
|
private boolean mAnimationAvailable;
|
|
private boolean mVideoReady;
|
|
private boolean mScrolling;
|
|
private BitmapDrawable mPreviewImage;
|
|
|
|
public GesturePreference(Context context, AttributeSet attrs) {
|
|
super(context, attrs);
|
|
mContext = context;
|
|
setLayoutResource(R.layout.gesture_preference);
|
|
TypedArray attributes = context.getTheme().obtainStyledAttributes(
|
|
attrs,
|
|
R.styleable.GesturePreference,
|
|
0, 0);
|
|
try {
|
|
int animation = attributes.getResourceId(R.styleable.GesturePreference_animation, 0);
|
|
mVideoPath = new Uri.Builder().scheme(ContentResolver.SCHEME_ANDROID_RESOURCE)
|
|
.authority(context.getPackageName())
|
|
.appendPath(String.valueOf(animation))
|
|
.build();
|
|
mMediaPlayer = MediaPlayer.create(mContext, mVideoPath);
|
|
if (mMediaPlayer != null) {
|
|
mMediaPlayer.setOnSeekCompleteListener(new MediaPlayer.OnSeekCompleteListener() {
|
|
@Override
|
|
public void onSeekComplete(MediaPlayer mp) {
|
|
mVideoReady = true;
|
|
}
|
|
});
|
|
|
|
mMediaPlayer.setOnPreparedListener(new MediaPlayer.OnPreparedListener() {
|
|
@Override
|
|
public void onPrepared(MediaPlayer mediaPlayer) {
|
|
mediaPlayer.setLooping(true);
|
|
}
|
|
});
|
|
}
|
|
mAnimationAvailable = true;
|
|
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Animation resource not found. Will not show animation.");
|
|
} finally {
|
|
attributes.recycle();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onBindViewHolder(PreferenceViewHolder holder) {
|
|
super.onBindViewHolder(holder);
|
|
final TextureView video = (TextureView) holder.findViewById(R.id.gesture_video);
|
|
final ImageView imageView = (ImageView) holder.findViewById(R.id.gesture_image);
|
|
final ImageView playButton = (ImageView) holder.findViewById(R.id.gesture_play_button);
|
|
final View animationFrame = holder.findViewById(R.id.gesture_animation_frame);
|
|
|
|
if (!mAnimationAvailable) {
|
|
animationFrame.setVisibility(View.GONE);
|
|
return;
|
|
}
|
|
|
|
video.setOnClickListener(new View.OnClickListener() {
|
|
@Override
|
|
public void onClick(View v) {
|
|
if (mMediaPlayer != null) {
|
|
if (mMediaPlayer.isPlaying()) {
|
|
mMediaPlayer.pause();
|
|
playButton.setVisibility(View.VISIBLE);
|
|
} else {
|
|
mMediaPlayer.start();
|
|
playButton.setVisibility(View.GONE);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
video.setSurfaceTextureListener(new TextureView.SurfaceTextureListener() {
|
|
@Override
|
|
public void onSurfaceTextureAvailable(SurfaceTexture surfaceTexture, int width,
|
|
int height) {
|
|
if (mMediaPlayer != null) {
|
|
mMediaPlayer.setSurface(new Surface(surfaceTexture));
|
|
mVideoReady = false;
|
|
mMediaPlayer.seekTo(0);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onSurfaceTextureSizeChanged(SurfaceTexture surfaceTexture, int width,
|
|
int height) {
|
|
}
|
|
|
|
@Override
|
|
public boolean onSurfaceTextureDestroyed(SurfaceTexture surfaceTexture) {
|
|
if (mPreviewImage != null && imageView.getDrawable() == null) {
|
|
imageView.setImageDrawable(mPreviewImage);
|
|
}
|
|
imageView.setVisibility(View.VISIBLE);
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onSurfaceTextureUpdated(SurfaceTexture surfaceTexture) {
|
|
if (mVideoReady && imageView.getVisibility() == View.VISIBLE) {
|
|
imageView.setVisibility(View.GONE);
|
|
} else if (mScrolling) {
|
|
mScrolling = false;
|
|
if (mMediaPlayer != null && mMediaPlayer.isPlaying()) {
|
|
mMediaPlayer.pause();
|
|
playButton.setVisibility(View.VISIBLE);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
if (mPreviewImage != null) {
|
|
imageView.setImageDrawable(mPreviewImage);
|
|
}
|
|
|
|
}
|
|
|
|
@Override
|
|
public void onDetached() {
|
|
if (mMediaPlayer != null) {
|
|
mMediaPlayer.stop();
|
|
mMediaPlayer.reset();
|
|
mMediaPlayer.release();
|
|
}
|
|
super.onDetached();
|
|
}
|
|
|
|
void setScrolling(boolean scrolling) {
|
|
mScrolling = scrolling;
|
|
}
|
|
|
|
void loadPreview(LoaderManager manager, int id) {
|
|
Loader<Bitmap> loader = manager.initLoader(id, Bundle.EMPTY, this);
|
|
}
|
|
|
|
void onViewVisible() {
|
|
if (mVideoReady && mMediaPlayer != null && !mMediaPlayer.isPlaying()) {
|
|
mMediaPlayer.seekTo(0);
|
|
}
|
|
}
|
|
|
|
private static final class PreviewRetriever extends AsyncLoader<Bitmap> {
|
|
private Uri mVideoPath;
|
|
|
|
public PreviewRetriever(Context context, Uri videoPath) {
|
|
super(context);
|
|
mVideoPath = videoPath;
|
|
}
|
|
|
|
@Override
|
|
public Bitmap loadInBackground() {
|
|
MediaMetadataRetriever mediaMetadata = new MediaMetadataRetriever();
|
|
try {
|
|
mediaMetadata.setDataSource(getContext(), mVideoPath);
|
|
return mediaMetadata.getFrameAtTime(0);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Unable to get animation preview.");
|
|
} finally {
|
|
mediaMetadata.release();
|
|
}
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public void onDiscardResult(final Bitmap result) {
|
|
if (result != null && !result.isRecycled()) {
|
|
result.recycle();
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
@Override
|
|
public Loader<Bitmap> onCreateLoader(int id, Bundle args) {
|
|
return new PreviewRetriever(mContext, mVideoPath);
|
|
}
|
|
|
|
@Override
|
|
public void onLoadFinished(final Loader<Bitmap> loader, final Bitmap bitmap) {
|
|
if (bitmap != null) {
|
|
mPreviewImage = new BitmapDrawable(mContext.getResources(), bitmap);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onLoaderReset(Loader<Bitmap> loader) {
|
|
}
|
|
|
|
}
|