Android 解决ViewPager.setCurrentItem中间很多页面切换问题

最近项目有个需求是主页包含4个Fragment,但还要求有滑动效果。相信大家第一个就会想到viewPager+fragment进行实现该功能了。 在做完功能后,发现有个问题,就是当我点击底部tab时,出现多个fragment依次闪过。比如我现在在第一个fragment,然后我点击第四个tab,页面就会经过第二个、第三个然后才到显示第四个。就是中间有个滑动的过程看起来很不爽。现在需要把这个滑动的时间去掉。

1、切换fragment就会用到 viewPager.setCurrentItem(int position); 部分源码如下

    /**
     * Set the currently selected page. If the ViewPager has already been through its first
     * layout with its current adapter there will be a smooth animated transition between
     * the current item and the specified item.
     *
     * @param item Item index to select
     */
    public void setCurrentItem(int item) {
        mPopulatePending = false;
        setCurrentItemInternal(item, !mFirstLayout, false);
    }

    /**
     * Set the currently selected page.
     *
     * @param item Item index to select
     * @param smoothScroll True to smoothly scroll to the new item, false to transition immediately
     */
    public void setCurrentItem(int item, boolean smoothScroll) {
        mPopulatePending = false;
        setCurrentItemInternal(item, smoothScroll, false);
    }

2、根据源码知道,ViewPager滑动使用的是Scroll,我们把Scroll的滑动时间duration 设置为0。那我们先自定义一个ViewPager

public class WViewPager extends ViewPager {
    
    private WViewPageHelper helper;
    
    public WViewPager(Context context) {
        this(context, null);
    }
    
    public WViewPager(Context context, AttributeSet attrs) {
        super(context, attrs);
        helper = new WViewPageHelper(this);
    }

    @Override
    public void setCurrentItem(int item) {
        setCurrentItem(item, true);
    }

    @Override
    public void setCurrentItem(int item, boolean smoothScroll) {
        WScroller scroller = helper.getScroller();
        //这里设置大于等于1,意思是2个相邻的tab点击也不出现闪动。可以根据需求自定义
        if (Math.abs(getCurrentItem() - item) >= 1) {
            scroller.setNoDuration(true);
            super.setCurrentItem(item, smoothScroll);
          //这里的设置false为切换后恢复默认的滑动状态,如果取消下面代码,当你滑动viewPager时,就不会有滑动切换效果了。
            scroller.setNoDuration(false);
        } else {
            scroller.setNoDuration(false);
            super.setCurrentItem(item, smoothScroll);
        }
    }
}

3、另外附上相关类代码。

public class WViewPageHelper {

    ViewPager viewPager;
    WScroller scroller;

    public WViewPageHelper(ViewPager viewPager) {
        this.viewPager = viewPager;
        init();
    }

    public WScroller getScroller() {
        return scroller;
    }
    private void init() {
        scroller = new WScroller(viewPager.getContext());
        Class<ViewPager> cl = ViewPager.class;
        try {
            //利用反射设置mScroller域为自己定义的WScroller,这里的命名(‘mScroller’)不能随意改
            Field field = cl.getDeclaredField("mScroller");
            field.setAccessible(true);
            field.set(viewPager, scroller);
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}
public class WScroller extends Scroller {
    private static final Interpolator sInterpolator = new Interpolator() {
        public float getInterpolation(float t) {
            t -= 1.0f;
            return t * t * t * t * t + 1.0f;
        }
    };
    public boolean noDuration;

    public void setNoDuration(boolean noDuration) {
        this.noDuration = noDuration;
    }

    public WScroller(Context context) {
        this(context, sInterpolator);
    }

    public WScroller(Context context, Interpolator interpolator) {
        super(context, interpolator);
    }

    @Override
    public void startScroll(int startX, int startY, int dx, int dy, int duration) {
        if (noDuration)
            //界面滑动不需要时间间隔
            super.startScroll(startX, startY, dx, dy, 0);
        else
            super.startScroll(startX, startY, dx, dy, duration);
    }
}

4、最后,效果图就暂时不贴了。亲测可用!

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值