Скорость замедленного просмотра контроллера Viewpager в android
Есть ли способ замедлить скорость прокрутки с помощью адаптера viewpager в андроиде?
Знаешь, я смотрел этот код. Я не могу понять, что я делаю неправильно.
try{
Field mScroller = mPager.getClass().getDeclaredField("mScroller");
mScroller.setAccessible(true);
Scroller scroll = new Scroller(cxt);
Field scrollDuration = scroll.getClass().getDeclaredField("mDuration");
scrollDuration.setAccessible(true);
scrollDuration.set(scroll, 1000);
mScroller.set(mPager, scroll);
}catch (Exception e){
Toast.makeText(cxt, "something happened", Toast.LENGTH_LONG).show();
}
Это ничего не меняет, но исключений не происходит?
Ответы
Ответ 1
Я начал с кода HighFlyer, который действительно изменил поле mScroller (что является хорошим началом), но не помог продлить продолжительность прокрутки, потому что ViewPager явно передает продолжительность mScroller при запросе прокрутки.
Расширение ViewPager не работает, так как важный метод (smoothScrollTo) не может быть переопределен.
Я решил исправить это, расширив Scroller следующим кодом:
public class FixedSpeedScroller extends Scroller {
private int mDuration = 5000;
public FixedSpeedScroller(Context context) {
super(context);
}
public FixedSpeedScroller(Context context, Interpolator interpolator) {
super(context, interpolator);
}
public FixedSpeedScroller(Context context, Interpolator interpolator, boolean flywheel) {
super(context, interpolator, flywheel);
}
@Override
public void startScroll(int startX, int startY, int dx, int dy, int duration) {
// Ignore received duration, use fixed one instead
super.startScroll(startX, startY, dx, dy, mDuration);
}
@Override
public void startScroll(int startX, int startY, int dx, int dy) {
// Ignore received duration, use fixed one instead
super.startScroll(startX, startY, dx, dy, mDuration);
}
}
И используя это так:
try {
Field mScroller;
mScroller = ViewPager.class.getDeclaredField("mScroller");
mScroller.setAccessible(true);
FixedSpeedScroller scroller = new FixedSpeedScroller(mPager.getContext(), sInterpolator);
// scroller.setFixedDuration(5000);
mScroller.set(mPager, scroller);
} catch (NoSuchFieldException e) {
} catch (IllegalArgumentException e) {
} catch (IllegalAccessException e) {
}
Я в основном жестко запрограммировал продолжительность до 5 секунд и заставил свой ViewPager использовать ее.
Надеюсь это поможет.
Ответ 2
Я хотел сделать сам и добился решения (используя отражение, однако). Он похож на принятое решение, но использует один и тот же интерполятор и только изменяет продолжительность, основанную на коэффициенте. Вам нужно использовать ViewPagerCustomDuration
в вашем XML вместо ViewPager
, а затем вы можете сделать это:
ViewPagerCustomDuration vp = (ViewPagerCustomDuration) findViewById(R.id.myPager);
vp.setScrollDurationFactor(2); // make the animation twice as slow
ViewPagerCustomDuration.java
import android.content.Context;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.view.animation.Interpolator;
import java.lang.reflect.Field;
public class ViewPagerCustomDuration extends ViewPager {
public ViewPagerCustomDuration(Context context) {
super(context);
postInitViewPager();
}
public ViewPagerCustomDuration(Context context, AttributeSet attrs) {
super(context, attrs);
postInitViewPager();
}
private ScrollerCustomDuration mScroller = null;
/**
* Override the Scroller instance with our own class so we can change the
* duration
*/
private void postInitViewPager() {
try {
Class<?> viewpager = ViewPager.class;
Field scroller = viewpager.getDeclaredField("mScroller");
scroller.setAccessible(true);
Field interpolator = viewpager.getDeclaredField("sInterpolator");
interpolator.setAccessible(true);
mScroller = new ScrollerCustomDuration(getContext(),
(Interpolator) interpolator.get(null));
scroller.set(this, mScroller);
} catch (Exception e) {
}
}
/**
* Set the factor by which the duration will change
*/
public void setScrollDurationFactor(double scrollFactor) {
mScroller.setScrollDurationFactor(scrollFactor);
}
}
ScrollerCustomDuration.java
import android.annotation.SuppressLint;
import android.content.Context;
import android.view.animation.Interpolator;
import android.widget.Scroller;
public class ScrollerCustomDuration extends Scroller {
private double mScrollFactor = 1;
public ScrollerCustomDuration(Context context) {
super(context);
}
public ScrollerCustomDuration(Context context, Interpolator interpolator) {
super(context, interpolator);
}
@SuppressLint("NewApi")
public ScrollerCustomDuration(Context context, Interpolator interpolator, boolean flywheel) {
super(context, interpolator, flywheel);
}
/**
* Set the factor by which the duration will change
*/
public void setScrollDurationFactor(double scrollFactor) {
mScrollFactor = scrollFactor;
}
@Override
public void startScroll(int startX, int startY, int dx, int dy, int duration) {
super.startScroll(startX, startY, dx, dy, (int) (duration * mScrollFactor));
}
}
Надеюсь, это поможет кому-то!
Ответ 3
Я нашел лучшее решение на основе @df778899 answer и
Android ValueAnimator API. Он отлично работает без отражения и очень гибкий.
Также нет необходимости создавать пользовательский ViewPager и помещать его в пакет android.support.v4.view.
Вот пример:
private void animatePagerTransition(final boolean forward) {
ValueAnimator animator = ValueAnimator.ofInt(0, viewPager.getWidth() - ( forward ? viewPager.getPaddingLeft() : viewPager.getPaddingRight() ));
animator.addListener(new Animator.AnimatorListener() {
@Override
public void onAnimationStart(Animator animation) {
}
@Override
public void onAnimationEnd(Animator animation) {
viewPager.endFakeDrag();
}
@Override
public void onAnimationCancel(Animator animation) {
viewPager.endFakeDrag();
}
@Override
public void onAnimationRepeat(Animator animation) {
}
});
animator.setInterpolator(new AccelerateInterpolator());
animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
private int oldDragPosition = 0;
@Override
public void onAnimationUpdate(ValueAnimator animation) {
int dragPosition = (Integer) animation.getAnimatedValue();
int dragOffset = dragPosition - oldDragPosition;
oldDragPosition = dragPosition;
viewPager.fakeDragBy(dragOffset * (forward ? -1 : 1));
}
});
animator.setDuration(AppConstants.PAGER_TRANSITION_DURATION_MS);
viewPager.beginFakeDrag();
animator.start();
}
Ответ 4
Как вы можете видеть в источниках ViewPager, длительность перехода, контролируемая объектом mScroller.
В documantation мы можем прочитать:
Продолжительность прокрутки может быть передана в конструкторе и задается максимальное время, в течение которого должна выполняться анимация прокрутки
Итак, если вы хотите контролировать скорость, вы можете изменить объект mScroller через отражение.
Вы должны написать что-то вроде этого:
setContentView(R.layout.main);
mPager = (ViewPager)findViewById(R.id.view_pager);
Field mScroller = ViewPager.class.getDeclaredField("mScroller");
mScroller.setAccessible(true);
mScroller.set(mPager, scroller); // initialize scroller object by yourself
Ответ 5
Это не идеальное решение, вы не можете сделать скорость медленнее, потому что это int
. Но для меня это достаточно медленно, и мне не нужно использовать рефлексию.
Обратите внимание на пакет, в котором находится класс. smoothScrollTo
имеет видимость пакета.
package android.support.v4.view;
import android.content.Context;
import android.util.AttributeSet;
public class SmoothViewPager extends ViewPager {
private int mVelocity = 1;
public SmoothViewPager(Context context) {
super(context);
}
public SmoothViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
void smoothScrollTo(int x, int y, int velocity) {
//ignore passed velocity, use one defined here
super.smoothScrollTo(x, y, mVelocity);
}
}
Ответ 6
Методы fakeDrag на ViewPager, похоже, предоставляют альтернативное решение.
Например, это будет страница с пункта 0 до 1:
rootView.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
ViewPager pager = (ViewPager) getActivity().findViewById(R.id.pager);
//pager.setCurrentItem(1, true);
pager.beginFakeDrag();
Handler handler = new Handler();
handler.post(new PageTurner(handler, pager));
}
});
private static class PageTurner implements Runnable {
private final Handler handler;
private final ViewPager pager;
private int count = 0;
private PageTurner(Handler handler, ViewPager pager) {
this.handler = handler;
this.pager = pager;
}
@Override
public void run() {
if (pager.isFakeDragging()) {
if (count < 20) {
count++;
pager.fakeDragBy(-count * count);
handler.postDelayed(this, 20);
} else {
pager.endFakeDrag();
}
}
}
}
(count * count
только там, чтобы ускорить ускорение)
Ответ 7
Я использовал
DecelerateInterpolator()
Вот пример:
mViewPager = (ViewPager) findViewById(R.id.container);
mViewPager.setAdapter(mSectionsPagerAdapter);
Field mScroller = null;
try {
mScroller = ViewPager.class.getDeclaredField("mScroller");
mScroller.setAccessible(true);
Scroller scroller = new Scroller(this, new DecelerateInterpolator());
mScroller.set(mViewPager, scroller);
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
Ответ 8
Вот ответ, использующий совершенно другой подход. Кто-то может сказать, что это хакерское чувство; однако он не использует рефлексию, и я бы сказал, что он всегда будет работать.
Мы находим следующий код внутри ViewPager.smoothScrollTo
:
if (velocity > 0) {
duration = 4 * Math.round(1000 * Math.abs(distance / velocity));
} else {
final float pageWidth = width * mAdapter.getPageWidth(mCurItem);
final float pageDelta = (float) Math.abs(dx) / (pageWidth + mPageMargin);
duration = (int) ((pageDelta + 1) * 100);
}
duration = Math.min(duration, MAX_SETTLE_DURATION);
Он вычисляет продолжительность, основанную на нескольких вещах. Видите все, что мы можем контролировать? mAdapter.getPageWidth
. Позвольте реализовать ViewPager.OnPageChangeListener в нашем адаптере. Мы собираемся обнаружить, когда ViewPager прокручивается, и дать поддельное значение для ширины. Если я хочу, чтобы продолжительность была k*100
, тогда я вернусь 1.0/(k-1)
для getPageWidth
. Ниже приведена Kotlin impl этой части адаптера, которая превращает продолжительность в 400:
var scrolling = false
override fun getPageWidth(position: Int): Float {
return if (scrolling) 0.333f else 1f
}
// OnPageChangeListener to detect scroll state
override fun onPageScrollStateChanged(state: Int) {
scrolling = state == ViewPager.SCROLL_STATE_SETTLING
}
override fun onPageScrolled(position: Int, positionOffset: Float, positionOffsetPixels: Int) {
}
override fun onPageSelected(position: Int) {
}
Не забудьте добавить адаптер как OnPageChangedListener.
В моем конкретном случае можно с уверенностью предположить, что пользователь не может перетащить между страницами. Если вам нужно поддерживать настройку после перетаскивания, вам нужно сделать немного больше в своих расчетах.
Один недостаток заключается в том, что это зависит от того, какое твердое значение 100
значение базы в ViewPager. Если это изменится, ваши длительности будут меняться с помощью этого подхода.
Ответ 9
binding.vpTour.beginFakeDrag();
lastFakeDrag = 0;
ValueAnimator va = ValueAnimator.ofInt(0, binding.vpTour.getWidth());
va.setDuration(1000);
va.addUpdateListener(animation -> {
if (binding.vpTour.isFakeDragging()) {
int animProgress = (Integer) animation.getAnimatedValue();
binding.vpTour.fakeDragBy(lastFakeDrag - animProgress);
lastFakeDrag = animProgress;
}
});
va.addListener(new AnimatorListenerAdapter() {
@Override
public void onAnimationEnd(Animator animation) {
super.onAnimationEnd(animation);
if (binding.vpTour.isFakeDragging()) {
binding.vpTour.endFakeDrag();
}
}
});
va.start();
Я хотел решить ту же проблему, что и все вы, и это мое решение для той же проблемы, но я считаю, что это решение более гибко, поскольку вы можете изменить продолжительность, как вам нравится, а также изменить интерполяцию значения по желанию для достижения различных визуальных эффектов.
Мое решение прокручивается со страницы "1" на страницу "2", поэтому только в увеличении позиций, но его можно легко изменить, чтобы перейти в уменьшающиеся позиции, выполнив "animProgress - lastFakeDrag" вместо "lastFakeDrag - animProgress".
Я считаю, что это наиболее гибкое решение для выполнения этой задачи.
Ответ 10
На основе принятого решения я создал класс kotlin с расширением для просмотра пейджера. Наслаждайтесь! :)
class ViewPageScroller : Scroller {
var fixedDuration = 1500 //time to scroll in milliseconds
constructor(context: Context) : super(context)
constructor(context: Context, interpolator: Interpolator) : super(context, interpolator)
constructor(context: Context, interpolator: Interpolator, flywheel: Boolean) : super(context, interpolator, flywheel)
override fun startScroll(startX: Int, startY: Int, dx: Int, dy: Int, duration: Int) {
// Ignore received duration, use fixed one instead
super.startScroll(startX, startY, dx, dy, fixedDuration)
}
override fun startScroll(startX: Int, startY: Int, dx: Int, dy: Int) {
// Ignore received duration, use fixed one instead
super.startScroll(startX, startY, dx, dy, fixedDuration)
}
}
fun ViewPager.setViewPageScroller(viewPageScroller: ViewPageScroller) {
try {
val mScroller: Field = ViewPager::class.java.getDeclaredField("mScroller")
mScroller.isAccessible = true
mScroller.set(this, viewPageScroller)
} catch (e: NoSuchFieldException) {
} catch (e: IllegalArgumentException) {
} catch (e: IllegalAccessException) {
}
}