it-swarm.com.ru

Остановка Exoplayer на экране ViewPager

Я использую ViewPager с одним экземпляром фрагмента, в котором я показываю Media файлы, такие как изображения, видео, аудио. 

Я реализовал ExoPlayer для обработки файлов Video & Audio. И Glide для изображений. 

Из этого чтобы избежать утечек памяти, я освобождаю объект ExoPlayer, например, в ItemViewerFragment.Java

 private void releasePlayer() {
        if (player != null) {
            player.release();
            player = null;
            trackSelector = null;
            simpleExoPlayerView.setPlayer(null);
        }
    }

    @Override
    public void onStart() {
        super.onStart();
        if (player == null && currentGalleryModel != null) {
            initializePlayer();
        }
    }

    @Override
    public void onResume() {
        super.onResume();
        if (player == null && currentGalleryModel != null) {
            initializePlayer();
        }
    }

    @Override
    public void onPause() {
        super.onPause();
        releasePlayer();
    }

    @Override
    public void onStop() {
        super.onStop();
        releasePlayer();
    }

И в onViewCreated() я инициализирую представление следующим образом:

 @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

        if (currentGalleryModel.isVideo() || currentGalleryModel.isAudio()) {

            simpleExoPlayerView.setVisibility(View.VISIBLE);
            imageView.setVisibility(View.GONE);

            initializePlayer();
        } else if (currentGalleryModel.isImage() || currentGalleryModel.isGif()) {
            simpleExoPlayerView.setVisibility(View.GONE);
            imageView.setVisibility(View.VISIBLE);


            Glide.with(this)
                    .load(currentGalleryModel.getFilePath())
                    .placeholder(Android.R.color.black)
                    .fitCenter()
                    .diskCacheStrategy(DiskCacheStrategy.NONE)
                    .into(imageView);

        }
    }

Я использую FragmentStatePagerAdapeter. Это метод getItem:

@Override
public Fragment getItem(int position) {
    return ItemViewerFragment.newInstance(mItems.get(position));
}

Я не могу обнаружить onPause фрагмента при первом пролистывании Viewpager. При втором пролистывании video/audio файлы перестают воспроизводиться.

В деятельности я попытался добавить .addOnPageChangeListener:

@Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {

     try {
        ((ItemViewerFragment)mAdapter.getItem(mPreviousPos)).imHiddenNow();
        } catch (Exception e) {
            e.printStackTrace();
        }
        mPreviousPos = position;
    }

И в ItemViewerFragment.Java:

public void imHiddenNow(){
      releasePlayer();
    }

Тем не менее video/audio продолжает играть.

Вот Видео Ссылка на Screencast.

Демонстрационный проект GitHub link .

Пожалуйста, обратите внимание, что решение, опубликованное ниже не работает.

10
Nilesh Deokar

Вот код для адаптера пейджера: 

 class ViewPagerAdapter extends FragmentPagerAdapter {
    private final List<Fragment> mFragmentList = new ArrayList<>();
    private final List<String> mFragmentTitleList = new ArrayList<>();

    public ViewPagerAdapter(FragmentManager manager) {
        super(manager);
    }

    @Override
    public Fragment getItem(int position) {
        return mFragmentList.get(position);
    }

    @Override
    public int getCount() {
        return mFragmentList.size();
    }

    public void addFragment(Fragment fragment, String title) {
        mFragmentList.add(fragment);
        mFragmentTitleList.add(title);
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return mFragmentTitleList.get(position);
    }
}

Вот слушатель прокрутки:

  viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
        @Override
        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
          //Stop media here.
        }

        @Override
        public void onPageSelected(int position) {
          //Save your previous position here.
        }

        @Override
        public void onPageScrollStateChanged(int state) {

        }
    });

Для мультимедиа вы можете использовать цикл Loop и сразу добавить все фрагменты в список, а затем использовать это для эффективности:

viewPager.setOffscreenPageLimit(3);

Это обеспечит доступность только 3 экземпляров вашего фрагмента, что достаточно.

Для использования одного фрагмента я бы предложил вам сделать это так:

 public MyFragment() {

}

//This is to send a file to the fragment if you need it.
public static MyFragment newInstance(File file) {
    MyFragment fragment = new MyFragment();
    Bundle bundle = new Bundle();
    bundle.putSerializable("file", file);
    fragment.setArguments(bundle);
    return fragment;
}

Затем в onCreate of Fragment вы можете получить ваш файл следующим образом:

File file;
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    file = getArguments().getSerializable("file");

}

Теперь добавьте свои фрагменты в пейджер так:

 for (int i = 0; i < totalFiles; i++) {
            viewPagerAdapter.addFragment(MyFragment.newInstance(fileList.get(i));
        }

Надеюсь это поможет.

3
Sarthak Gandhi

Я придерживался подхода поддержания HashMap фрагментированных объектов внутри PagerAdapter 

  1. Объявите интерфейс:

`

interface FragmentLifecycle {
    void onPauseFragment()
}
  1. Реализуйте интерфейс во Фрагменте.

`

public void onPauseFragment() {
            if (simpleExoPlayer != null){        simpleExoPlayer.setPlayWhenReady(false);
      }
 }

`

  1. Сохраните все объекты фрагмента в HashMap<Integer,Fragment> с соответствующими позициями в качестве ключа. Объявите hashmap внутри PagerAdapter. Также объявите один метод получения для доступа к объектам фрагмента из hashmap. например.

    @Override
    public Fragment getItem(int position) {
        ItemViewerFragment fragment = ItemViewerFragment.newInstance(mItems.get(position));
        mFragments.put(position,fragment);
        return fragment;
    }
    

    public ItemViewerFragment getMapItem (int position) { вернуть mFragments.get (position); }

  2. В деятельности, в которой вы объявили viewPager, оставьте одну переменную currentPosition и внедрите ViewPager.OnPageChangeListener

  3. Внутри метода onPageSelected

        @Override
        public void onPageSelected(int position) {
            if(mAdapter.getMapItem(currentPosition) != null) (mAdapter.getMapItem(currentPosition)).onPauseFragment();
            currentPosition = position;
        }
    
1
Nilesh Deokar

Проблема в том, что onPause и onResume не вызываются, когда видимость фрагмента изменилась в ViewPager . Решение состоит в том, чтобы добавить 2 события видимости: losingVisibility и gainVisibility.

Почему это отличное решение?

Поскольку вы сохраняете инфраструктуру, управляющую кэшем фрагмента и жизненным циклом . Мы просто добавляем обратные вызовы, необходимые для приостановки и возобновления наших медиафайлов в нашем фрагменте.

Шаг за шагом:

Приведенный ниже код является только объяснением моего кода. Проверьте Step * .Java classes, чтобы увидеть полную реализацию.

  1. Создайте методы losingVisibility и gainVisibility в YourFragment.Java :

    public class YourFragment extends Fragment {
    
        /**
          * This method is only used by viewpager because the viewpager doesn't call onPause after
          * changing the fragment
          */
        public void losingVisibility() {
            // IMPLEMENT YOUR PAUSE CODE HERE
            savePlayerState();
            releasePlayer();
        }
    
        /**
          * This method is only used by viewpager because the viewpager doesn't call onPause after
          * changing the fragment
        */
        public void gainVisibility() {
            // IMPLEMENT YOUR RESUME CODE HERE
            loadVideo();
        }
    }
    
  2. Вызывайте losingVisibility и gainVisibility каждый раз, когда выбирается новая страница (onPageSelected) в YourActivity.Java :

    mViewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
        @Override
        public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        }
    
        @Override
        public void onPageSelected(int position) {
            YourFragment cachedFragmentLeaving = mYourPagerAdapter.getCachedItem(mCurrentItem);
            if (cachedFragmentLeaving != null) {
                cachedFragmentLeaving.losingVisibility();
            }
            mCurrentItem = position;
            YourFragment cachedFragmentEntering = mYourPagerAdapter.getCachedItem(mCurrentItem);
            if (cachedFragmentEntering != null) {
                cachedFragmentEntering.gainVisibility();
            }
        }
    
        @Override
        public void onPageScrollStateChanged(int state) {
        }
    });
    
  3. Добавьте getCachedItem в YourPagerAdapter.Java :

3-й шаг - добавление метода для извлечения кэшированных фрагментов. Для этого мы должны кэшировать ссылку на созданный фрагмент (переопределение instantiateItem) и освободить ту же ссылку (переопределение destroyItem).

    public class YourPagerAdapter extends FragmentStatePagerAdapter {

            private SparseArray<YourFragment> mFragmentsHolded = new SparseArray<>();


        @NonNull
        @Override
        public Object instantiateItem(ViewGroup container, int position) {
            Object fragment = super.instantiateItem(container, position);
            if(fragment instanceof StepFragment) {
                mFragmentsHolded.append(position, (StepFragment) fragment);
            }
            return fragment;
        }

        @Override
        public void destroyItem(ViewGroup container, int position, Object object) {
            mFragmentsHolded.delete(position);
            super.destroyItem(container, position, object);
        }

        public YourFragment getCachedItem(int position) {
            return mFragmentsHolded.get(position, null);
        }
    }
1
Erick M. Sprengel

Прошло уже больше года с тех пор, как я использовал Exoplayer, и я решил подобную проблему. Обратите внимание, что API-интерфейсы немного изменились, поэтому возьмите следующий код, чтобы получить представление о том, как реализовать потенциальное решение. Пожалуйста, дайте мне знать, если это не сработает, я еще посмотрю на API и свяжусь с вами. 

Приходя к решению:

private int mPlayerCurrentPosition;

private int getCurrentPlayerPosition() {
     return mExoPlayer.getCurrentPosition();
}

// call this from onPause
private void releaseExoplayer() {
     mPlayerCurrentPosition = getPlayerCurrentPosition();
     mExoPlayer.setPlayWhenReady(false);
     mExoPlayer.release(); // this will make the player object eligible for GC
}

private void resumePlaybackFromPreviousPosition(int prevPosition) {
    mExoPlayer.seekTo(mPlayerCurrentPosition );
}
1
Umer Farooq

я знаю, что так поздно, но я надеюсь, что это кому-нибудь поможет ...
viewPager сохраняет фрагмент за кадром запущенным ..
так что у тебя есть: 
prev закадровый фрагмент (начало) -> видимый фрагмент (работает) -> следующий закадровый фрагмент (начало)
Решение в том, что вы можете переопределить функцию Fragment.setUserVisibleHint() и обработать видео воспроизведения/паузы.

0
Seif E. Attia