プレーヤーを実装しましたが、問題が発生しました。ビデオの再生中にアプリを閉じて再開すると、ビデオ画面がフリーズします。理解を深めるためにGoogleのExoPlayerデモアクティビティを見ましたが、アプリに実装するためにそれを通り抜けることができませんでした。ここにプレーヤーアクティビティを添付しました。完全なコードについては、使用するファイルの完全なセットのGitHubリポジトリを共有しています。
RecipeStepDetailFragment.Java
package com.example.Android.recipe.ui;
import Android.content.Context;
import Android.content.res.Configuration;
import Android.net.Uri;
import Android.os.Bundle;
import Android.os.Handler;
import Android.support.v4.app.Fragment;
import Android.support.v4.content.ContextCompat;
import Android.view.LayoutInflater;
import Android.view.View;
import Android.view.ViewGroup;
import Android.widget.Button;
import Android.widget.ImageView;
import Android.widget.LinearLayout;
import Android.widget.TextView;
import Android.widget.Toast;
import com.example.Android.recipe.R;
import com.example.Android.recipe.pojo.Recipe;
import com.example.Android.recipe.pojo.Step;
import com.google.Android.exoplayer2.LoadControl;
import com.google.Android.exoplayer2.SimpleExoPlayer;
import com.google.Android.exoplayer2.source.ExtractorMediaSource;
import com.google.Android.exoplayer2.ui.AspectRatioFrameLayout;
import com.google.Android.exoplayer2.ui.SimpleExoPlayerView;
import Java.util.ArrayList;
import Java.util.List;
import com.google.Android.exoplayer2.DefaultLoadControl;
import com.google.Android.exoplayer2.ExoPlayerFactory;
import com.google.Android.exoplayer2.extractor.DefaultExtractorsFactory;
import com.google.Android.exoplayer2.source.MediaSource;
import com.google.Android.exoplayer2.trackselection.AdaptiveVideoTrackSelection;
import com.google.Android.exoplayer2.trackselection.DefaultTrackSelector;
import com.google.Android.exoplayer2.trackselection.TrackSelection;
import com.google.Android.exoplayer2.upstream.BandwidthMeter;
import com.google.Android.exoplayer2.upstream.DefaultBandwidthMeter;
import com.google.Android.exoplayer2.upstream.DefaultDataSourceFactory;
import com.google.Android.exoplayer2.util.Util;
import com.squareup.picasso.Picasso;
import static com.example.Android.recipe.ui.RecipeActivity.SELECTED_INDEX;
import static com.example.Android.recipe.ui.RecipeActivity.SELECTED_RECIPES;
import static com.example.Android.recipe.ui.RecipeActivity.SELECTED_STEPS;
public class RecipeStepDetailFragment extends Fragment {
private SimpleExoPlayerView simpleExoPlayerView;
private SimpleExoPlayer player;
private BandwidthMeter bandwidthMeter;
private ArrayList<Step> steps = new ArrayList<>();
private int selectedIndex;
private Handler mainHandler;
ArrayList<Recipe> recipe;
String recipeName;
public RecipeStepDetailFragment() { }
private ListItemClickListener itemClickListener;
public interface ListItemClickListener {
void onListItemClick(List<Step> allSteps,int Index,String recipeName);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
TextView textView;
mainHandler = new Handler();
bandwidthMeter = new DefaultBandwidthMeter();
itemClickListener =(RecipeDetailActivity)getActivity();
recipe = new ArrayList<>();
if(savedInstanceState != null) {
steps = savedInstanceState.getParcelableArrayList(SELECTED_STEPS);
selectedIndex = savedInstanceState.getInt(SELECTED_INDEX);
recipeName = savedInstanceState.getString("Title");
}
else {
steps =getArguments().getParcelableArrayList(SELECTED_STEPS);
if (steps!=null) {
steps =getArguments().getParcelableArrayList(SELECTED_STEPS);
selectedIndex=getArguments().getInt(SELECTED_INDEX);
recipeName=getArguments().getString("Title");
}
else {
recipe =getArguments().getParcelableArrayList(SELECTED_RECIPES);
steps=(ArrayList<Step>)recipe.get(0).getSteps();
selectedIndex=0;
}
}
View rootView = inflater.inflate(R.layout.recipe_step_detail_fragment_body_part, container, false);
textView = (TextView) rootView.findViewById(R.id.recipe_step_detail_text);
textView.setText(steps.get(selectedIndex).getDescription());
textView.setVisibility(View.VISIBLE);
simpleExoPlayerView = (SimpleExoPlayerView) rootView.findViewById(R.id.playerView);
simpleExoPlayerView.setResizeMode(AspectRatioFrameLayout.RESIZE_MODE_FIT);
String videoURL = steps.get(selectedIndex).getVideoURL();
if (rootView.findViewWithTag("sw600dp-port-recipe_step_detail")!=null) {
recipeName=((RecipeDetailActivity) getActivity()).recipeName;
((RecipeDetailActivity) getActivity()).getSupportActionBar().setTitle(recipeName);
}
String imageUrl=steps.get(selectedIndex).getThumbnailURL();
if (imageUrl!="") {
Uri builtUri = Uri.parse(imageUrl).buildUpon().build();
ImageView thumbImage = (ImageView) rootView.findViewById(R.id.thumbImage);
Picasso.with(getContext()).load(builtUri).into(thumbImage);
}
if (!videoURL.isEmpty()) {
initializePlayer(Uri.parse(steps.get(selectedIndex).getVideoURL()));
if (rootView.findViewWithTag("sw600dp-land-recipe_step_detail")!=null) {
getActivity().findViewById(R.id.fragment_container2).setLayoutParams(new LinearLayout.LayoutParams(-1,-2));
simpleExoPlayerView.setResizeMode(AspectRatioFrameLayout.RESIZE_MODE_FIXED_WIDTH);
}
else if (isInLandscapeMode(getContext())){
textView.setVisibility(View.GONE);
}
}
else {
player=null;
simpleExoPlayerView.setForeground(ContextCompat.getDrawable(getContext(), R.drawable.ic_visibility_off_white_36dp));
simpleExoPlayerView.setLayoutParams(new LinearLayout.LayoutParams(300, 300));
}
Button mPrevStep = (Button) rootView.findViewById(R.id.previousStep);
Button mNextstep = (Button) rootView.findViewById(R.id.NeXTSTEP);
mPrevStep.setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
if (steps.get(selectedIndex).getId() > 0) {
if (player!=null){
player.stop();
}
itemClickListener.onListItemClick(steps,steps.get(selectedIndex).getId() - 1,recipeName);
}
else {
Toast.makeText(getActivity(),"You already are in the First step of the recipe", Toast.LENGTH_SHORT).show();
}
}});
mNextstep.setOnClickListener(new View.OnClickListener() {
public void onClick(View view) {
int lastIndex = steps.size()-1;
if (steps.get(selectedIndex).getId() < steps.get(lastIndex).getId()) {
if (player!=null){
player.stop();
}
itemClickListener.onListItemClick(steps,steps.get(selectedIndex).getId() + 1,recipeName);
}
else {
Toast.makeText(getContext(),"You already are in the Last step of the recipe", Toast.LENGTH_SHORT).show();
}
}});
return rootView;
}
private void initializePlayer(Uri mediaUri) {
if (player == null) {
TrackSelection.Factory videoTrackSelectionFactory = new AdaptiveVideoTrackSelection.Factory(bandwidthMeter);
DefaultTrackSelector trackSelector = new DefaultTrackSelector(mainHandler, videoTrackSelectionFactory);
LoadControl loadControl = new DefaultLoadControl();
player = ExoPlayerFactory.newSimpleInstance(getContext(), trackSelector, loadControl);
simpleExoPlayerView.setPlayer(player);
String userAgent = Util.getUserAgent(getContext(), "Baking App");
MediaSource mediaSource = new ExtractorMediaSource(mediaUri, new DefaultDataSourceFactory(getContext(), userAgent), new DefaultExtractorsFactory(), null, null);
player.prepare(mediaSource);
player.setPlayWhenReady(true);
}
}
@Override
public void onSaveInstanceState(Bundle currentState) {
super.onSaveInstanceState(currentState);
currentState.putParcelableArrayList(SELECTED_STEPS,steps);
currentState.putInt(SELECTED_INDEX,selectedIndex);
currentState.putString("Title",recipeName);
}
public boolean isInLandscapeMode( Context context ) {
return (context.getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE);
}
@Override
public void onDetach() {
super.onDetach();
if (player!=null) {
player.stop();
player.release();
}
}
@Override
public void onDestroyView() {
super.onDestroyView();
if (player!=null) {
player.stop();
player.release();
player=null;
}
}
@Override
public void onStop() {
super.onStop();
if (player!=null) {
player.stop();
player.release();
}
}
@Override
public void onPause() {
super.onPause();
if (player!=null) {
player.stop();
player.release();
}
}
}
完全なプロジェクトリポジトリ: https://github.com/mtp2697/Udacity-AndroidDeveloperNanodegree-BakingApp
OnPause()およびonResume()のビデオプレーヤーの状態の復元を支援します
ありがとう、PraveenThirumurugan。
一時停止時にプレーヤーの位置を保存できます。
_position = player.getCurrentPosition(); //then, save it on the bundle.
_
そして、それを復元するときに、そこにある場合は、次のことができます。
_if (position != C.TIME_UNSET) player.seekTo(position);
_
prepare()
メソッドのinitializePlayer()
の前。
わかりました。プロジェクトのクローンを作成して、機能させました。基本的に変更したのは次のとおりです。
前に言ったことを追加してから:
_position = C.TIME_UNSET;
if (savedInstanceState != null) {
//...your code...
position = savedInstanceState.getLong(SELECTED_POSITION, C.TIME_UNSET);
}
_
videoUriをグローバルにしました
_videoUri = Uri.parse(steps.get(selectedIndex).getVideoURL());
_
onResumeを追加:
_@Override
public void onResume() {
super.onResume();
if (videoUri != null)
initializePlayer(videoUri);
}
_
onPauseを更新:
_@Override
public void onPause() {
super.onPause();
if (player != null) {
position = player.getCurrentPosition();
player.stop();
player.release();
player = null;
}
}
_
そしてonSaveInstanceState:
_currentState.putLong(SELECTED_POSITION, position);
_
最後、onDetach
onDestroyView
onStop
を削除しました。
明らかに、これは「それを機能させるためだけに」であり、あなたはそれにもっと取り組む必要があります。
私はこれが古いスレッドであることを知っていますが、ここに私の修正があります
protected void onPause() {
player.setPlayWhenReady(false);
super.onPause();
}
protected void onResume() {
player.setPlayWhenReady(true);
super.onResume();
}
これにより、アクティビティの一時停止時にビデオが一時停止し、アクティビティの再開時に再開します。