请选择 进入手机版 | 继续访问电脑版

Android无限循环滚动

[复制链接]
为你演绎 发表于 2021-1-2 11:50:04 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题
传统的ViewPager做循环滚动有两种思路。

  • 一种是设置count为Integer.MAX,然后根据index对实际数量取模
  • 一种是在开头在开头添加end,在末了添加start。简单的说就是多两个,滑动到这两个的时候直接setCurrentItem到真正的位置。
在观察pdd的拼单的循环滚动的时候,想到几种实现方式。

  • 通过Recyclerview,同样跟ViewPager做循环滚动的思路类似,多一点要拦截掉所有的触摸事件。但是这种方式的话无法像pdd的效果那样设置进入和出去的动画。
  • 通过改造VerticalViewpager的形式,应该也是可以的,但是感觉比力贫苦。
  • 通过自界说的方式实现。(原本以为挺简单的,实现了下,代码不多但是有些小细节需要注意下。)
我选择了自界说的这里只是一个demo,提供一种思路。
最焦点的就是上面的item滑出屏幕的时候将它remove掉然后再加到自界说的ViewGroup的末了。
  1. public class LoopView extends ViewGroup {    private static final String TAG = "LoopView";    private float dis;    private ObjectAnimator animator;    private int currentIndex = 0;    private Handler handler = new Handler(Looper.getMainLooper());    public LoopView(Context context) {        super(context);        init();    }    public LoopView(Context context, AttributeSet attrs) {        super(context, attrs);        init();    }    public LoopView(Context context, AttributeSet attrs, int defStyleAttr) {        super(context, attrs, defStyleAttr);        init();    }    void init() {        View view1 = new View(getContext());        view1.setTag("gray");        view1.setBackgroundColor(Color.GRAY);        LayoutParams layoutParams = new LayoutParams(LayoutParams.MATCH_PARENT, 200);        addView(view1, layoutParams);        View view2 = new View(getContext());        view2.setTag("red");        view2.setBackgroundColor(Color.RED);        LayoutParams layoutParams1 = new LayoutParams(LayoutParams.MATCH_PARENT, 200);        addView(view2, layoutParams1);        View view3 = new View(getContext());        view3.setTag("green");        view3.setBackgroundColor(Color.GREEN);        LayoutParams layoutParams2 = new LayoutParams(LayoutParams.MATCH_PARENT, 200);        addView(view3, layoutParams2);        animator = ObjectAnimator.ofFloat(this, "dis", 0, 1);        animator.setDuration(2000);        animator.addListener(new AnimatorListenerAdapter() {            @Override            public void onAnimationEnd(Animator animation) {                currentIndex++;                View first = getChildAt(0);                removeView(first);                addView(first);                handler.postDelayed(new Runnable() {                    @Override                    public void run() {                        animator.clone().start();                    }                }, 3000);            }        });    }    public void start() {        animator.start();    }    @Override    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        measureChildren(widthMeasureSpec, heightMeasureSpec);        super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(200, MeasureSpec.EXACTLY));    }    @Override    protected void onLayout(boolean changed, int l, int t, int r, int b) {        int childCount = getChildCount();        int top = currentIndex * getMeasuredHeight();        for (int i = 0; i < childCount; i++) {            View childAt = getChildAt(i);            childAt.layout(l, top, r, top + childAt.getMeasuredHeight());            top += childAt.getMeasuredHeight();        }    }    public float getDis() {        return dis;    }    public void setDis(float dis) {        this.dis = dis;        float disY = dis * getHeight();        scrollTo(0, (int) (currentIndex * getHeight() + disY));    }}
复制代码
需要注意的就是onLayout的时候对于top的取值。

来源:https://blog.csdn.net/qq_28302795/article/details/112017591
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则


专注素材教程免费分享
全国免费热线电话

18768367769

周一至周日9:00-23:00

反馈建议

27428564@qq.com 在线QQ咨询

扫描二维码关注我们

Powered by Discuz! X3.4© 2001-2013 Comsenz Inc.( 蜀ICP备2021001884号-1 )