Как реализовать exoplayer с recyclerview и viewpager2? - PullRequest
0 голосов
/ 03 августа 2020

Я создаю видеопоток и использую VideoView для предварительного просмотра видео в viewpager2. Но поскольку это было медленно, я попытался переключиться на экзоплеер. Все работает нормально, но проблема в том, что каждое видео в адаптере загружается сразу, и когда я смахиваю из окна просмотра, видео не останавливается. Вот код для адаптера:

public class VideosAdapter extends RecyclerView.Adapter<VideosAdapter.VideoViewHolder> {

    private List<VideoItem> videoItems;

    public VideosAdapter(List<VideoItem> videoItems) {
        this.videoItems = videoItems;
    }

    @NonNull
    @Override
    public VideoViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        return new VideoViewHolder(
               LayoutInflater.from(parent.getContext()).inflate(
                       R.layout.item_container_video,
                       parent,
                       false
               )
        );
    }



    @Override
    public void onBindViewHolder(@NonNull VideoViewHolder holder, int position) {
      holder.setVideoData(videoItems.get(position));
    }

    

    @Override
    public int getItemCount() {
        return videoItems.size();
    }


    static class VideoViewHolder extends RecyclerView.ViewHolder {

        PlayerView playerView;
        TextView video__description;
        ProgressBar videoProgressBar;
        SimpleExoPlayer simpleExoPlayer;
        LoopingMediaSource loopingMediaSource;
        DefaultHttpDataSourceFactory factory;
        ExtractorsFactory extractorsFactory;
        BandwidthMeter bandwidthMeter;
        TrackSelector trackSelector;

        public VideoViewHolder(@NonNull View itemView) {
            super(itemView);
            playerView = itemView.findViewById(R.id.player__view);
            video__description = itemView.findViewById(R.id.video__description);
            videoProgressBar = itemView.findViewById(R.id.videoLoader);
            factory = new DefaultHttpDataSourceFactory("exoplayer_video");
            extractorsFactory = new DefaultExtractorsFactory();
            bandwidthMeter = new DefaultBandwidthMeter();
            trackSelector = new DefaultTrackSelector(
                    new AdaptiveTrackSelection.Factory(bandwidthMeter)
            );
        }

        void setVideoData(VideoItem videoItem) {
            Uri videoUrl =Uri.parse(videoItem.videoURL);
            simpleExoPlayer = ExoPlayerFactory.newSimpleInstance(videoItem.context, trackSelector);
            MediaSource mediaSource = new ExtractorMediaSource(videoUrl,factory,extractorsFactory,null,null);
            loopingMediaSource = new LoopingMediaSource(mediaSource);
            playerView.setPlayer(simpleExoPlayer);
            playerView.setKeepScreenOn(true);
            simpleExoPlayer.prepare(loopingMediaSource);
            simpleExoPlayer.setPlayWhenReady(true);
            video__description.setText(videoItem.videoDescription);
            simpleExoPlayer.addListener(new Player.DefaultEventListener() {
                @Override
                public void onPlayerStateChanged(boolean playWhenReady, int playbackState) {
                    super.onPlayerStateChanged(playWhenReady, playbackState);
                    if(playbackState==Player.STATE_BUFFERING) {
                        videoProgressBar.setVisibility(View.VISIBLE);
                    } else if(playbackState==Player.STATE_READY) {
                        videoProgressBar.setVisibility(View.GONE);
                    }
                }

                @Override
                public void onPlayerError(ExoPlaybackException error) {
                    super.onPlayerError(error);
                }
            });
        }
    }
}


Вот код для класса videoItems:

public class VideoItem {
    public String videoURL, videoDescription;
    public Context context;

    public VideoItem(String videoURL, String videoDescription, Context context) {
        this.videoURL = videoURL;
        this.videoDescription = videoDescription;
        this.context = context;
    }
}

А вот код для настройки адаптера:

public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {

        // Inflate the layout for this fragment
        View root = inflater.inflate(R.layout.fragment_first, container, false);


        final ViewPager2 videosViewPager = root.findViewById(R.id.videosViewPager);

            List<VideoItem> videoItems = new ArrayList<>();

VideoItem videoItemFirst = new VideoItem("https://www.infinityandroid.com/videos/video1.mp4","Hello World", requireActivity().getApplicationContext());
            videoItems.add(videoItemFirst);

            VideoItem videoItemSecond = new VideoItem("https://www.infinityandroid.com/videos/video2.mp4","Hello World", requireActivity().getApplicationContext());
            videoItems.add(videoItemSecond);

            VideoItem videoItemThird = new VideoItem("https://www.infinityandroid.com/videos/video3.mp4","Hello World", requireActivity().getApplicationContext());
            videoItems.add(videoItemThird);

videosViewPager.setAdapter(new VideosAdapter(videoItems));


return root;
}

Вот файл макета:


<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    tools:context=".fragments.firstFragment"
    >

    <androidx.viewpager2.widget.ViewPager2
        android:id="@+id/videosViewPager"
        android:orientation="horizontal"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toTopOf="parent"
        app:layout_constraintEnd_toStartOf="parent"
         />

</androidx.constraintlayout.widget.ConstraintLayout>

Кажется, загружаются все видео сразу, когда загружается фрагмент, и когда я смахиваю, чтобы перейти к следующему видео, предыдущее видео не сделать паузу. Как это исправить? Есть способ лучше? Надеюсь, ты ответишь.

...