红橙Darren视频笔记 仿汽车之家 可拖动列表

最终效果演示
在这里插入图片描述

1.ViewDragHelper简介

1.1ViewDragHelper可以做什么基本效果

在这里插入图片描述

1.2实现code

public class DragView extends FrameLayout {
    ViewDragHelper mViewDragHelper;

    public DragView(@NonNull Context context) {
        this(context, null);
    }

    public DragView(@NonNull Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DragView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        mViewDragHelper = ViewDragHelper.create(this, mDragHelperCallback);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        mViewDragHelper.processTouchEvent(event);//需要给mViewDragHelper投喂事件才有效果 否则没反应
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                //同理 需要给子view投喂事件 必须返回true
                return true;
        }
        return super.onTouchEvent(event);
    }

    private ViewDragHelper.Callback mDragHelperCallback = new ViewDragHelper.Callback() {
        @Override
        public boolean tryCaptureView(@NonNull View view, int i) {
            //返回true代表可以拖动 可以根据view来判断哪个view可以拖动
            return true;
        }

        @Override
        public int clampViewPositionVertical(View child, int top, int dy) {
            // top代表垂直拖动的距离 dy代表垂直方向的速度还是加速度之类的?
            Log.d("TAG", "clampViewPositionVertical: top->" + top + " dy-> " + dy);
            return top;
        }

        @Override
        public int clampViewPositionHorizontal(View child, int left, int dx) {
            // left代表水平拖动的距离 dx代表水平方向的速度还是加速度之类的?
            Log.d("TAG", "clampViewPositionVertical: left->" + left + " dx-> " + dx);
            return left;
        }
    };
}
<?xml version="1.0" encoding="utf-8"?>
<com.example.chj.draglistview.DragView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="400dp"
        android:background="#ccc">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="后面" />
    </RelativeLayout>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="400dp"
        android:background="#cfc">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="前面" />
    </RelativeLayout>

</com.example.chj.draglistview.DragView>

2.效果分析实现

我们需要基于上面的代码实现一个效果, 前面是list 可以拖动 后面是一个view不能拖动
大致效果如下:
其他需求拆分:
注意实现2.1至2.4的功能时先不要直接使用listview 先用其他view代替

2.1 后面不能拖动

2.2 列表只能垂直拖动

2.3 垂直拖动的范围最大只能是后面菜单 View 的高度

2.4 手指松开的时候两者选其一,要么打开要么关闭

3事件的分发和拦截

当我们将普通的view替换为listview时我们发现所有的事件都被listview吃掉了 之前我们都功能都不起作用了,这里就涉及到事件分发和拦截了:原因是listview内部调用requestDisallowInterceptTouchEvent请求父类不要拦截事件导致事件被listview的各个子项消耗了
那么我们需要分析,不是所有情况都是让listview获取事件的

3.1.菜单打开处理

(展开态-非折叠态)父容器拦截所有事件 listview不应该自己处理事件

3.2.菜单关闭处理

(折叠态)并且是向下滑动 并且listView内部不能向下滚动 父容器拦截所有事件 listview不应该自己处理事件
在处理这事件分发的环节遇到很多问题,需要停下来好好想想需求部分 需求清晰逻辑才清晰

4 主要代码部分

4.1自定义部分

public class DragView extends FrameLayout {
    private ViewDragHelper mViewDragHelper;
    private ViewGroup frontView, behindView;
    private boolean isMenuOpen = false;

    public DragView(@NonNull Context context) {
        this(context, null);
    }

    public DragView(@NonNull Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DragView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        //1.创建和使用ViewDragHelper
        mViewDragHelper = ViewDragHelper.create(this, mDragHelperCallback);
    }

    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();
        //注意赋值时机 不能太靠前 太早behindView和frontView都还是空
        //Log.d("TAG", "DragView: " + this.getChildCount());
        if (this.getChildCount() != 2) {
            throw new RuntimeException("DragView 必须又2个子view");
        }
        //固然可以用findViewById来定位各个view 但这种getChildAt的方式更有利于我们了解各个view的位置
        behindView = (ViewGroup) this.getChildAt(0);
        frontView = (ViewGroup) this.getChildAt(1);
    }

    ListView getFrontListView() {
        return (ListView) frontView;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        //1.创建和使用ViewDragHelper
        mViewDragHelper.processTouchEvent(event);//需要给mViewDragHelper投喂事件才有效果 否则没反应
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                //同理 需要给子view投喂事件 必须返回true
                return true;
        }
        return super.onTouchEvent(event);
    }

    private ViewDragHelper.Callback mDragHelperCallback = new ViewDragHelper.Callback() {
        @Override
        public boolean tryCaptureView(@NonNull View view, int i) {
            return view == frontView;//2.1 后面不能拖动 只有前面的view可以拖动
        }

        @Override
        public int clampViewPositionVertical(View child, int top, int dy) {
            //top代表垂直拖动的距离 dy代表垂直方向的速度还是加速度之类的?
            //Log.d("TAG", "clampViewPositionVertical: top->" + top + " dy-> " + dy);
            //2.3 垂直拖动的范围最大只能是后面菜单 View 的高度
            if (top < 0) {
                return 0;
            }
            if (top > behindView.getMeasuredHeight()) {
                return behindView.getMeasuredHeight();
            }
            return top;
        }

        @Override
        public int clampViewPositionHorizontal(View child, int left, int dx) {
            // left代表水平拖动的距离 dx代表水平方向的速度还是加速度之类的?
            //Log.d("TAG", "clampViewPositionHorizontal: left->" + left + " dx-> " + dx);
            return 0;//2.2 列表只能垂直拖动 注释掉clampViewPositionHorizontal或者强制返回0 让view无法水平拖动
        }

        @Override
        public void onViewReleased(@NonNull View releasedChild, float xvel, float yvel) {
            super.onViewReleased(releasedChild, xvel, yvel);
            //Log.d("TAG", "onViewReleased: xvel->" + xvel + " yvel-> " + yvel);
            //Log.d("TAG", "onViewReleased: releasedChild.getTop()->" + releasedChild.getTop());
            //2.4 手指松开的时候两者选其一,要么打开要么关闭 2选1 要么全露出后面的view 要么全遮住后面的view
            //2.4 需要结合computeScroll使用 两个都要调用刷新
            if (releasedChild.getTop() < behindView.getMeasuredHeight() / 2) {
                //全遮住后面的view 折叠态 菜单关闭
                isMenuOpen = false;
                mViewDragHelper.settleCapturedViewAt(0, 0);
            } else {
                //全露出后面的view 展开态 菜单打开
                isMenuOpen = true;
                mViewDragHelper.settleCapturedViewAt(0, behindView.getMeasuredHeight());
            }
            invalidate();
        }

    };

    /**
     * Called by a parent to request that a child update its values for mScrollX
     * and mScrollY if necessary. This will typically be done if the child is
     * animating a scroll using a {@link android.widget.Scroller Scroller}
     * object.
     */
    @Override
    public void computeScroll() {
        if (mViewDragHelper.continueSettling(true)) {
            invalidate();
        }
    }

    //Ignoring pointerId=0 because ACTION_DOWN was not received for this pointer before ACTION_MOVE.
    //It likely happened because  ViewDragHelper did not receive all the events in the event stream.
    //上面这个报错 原因如下
    //DragView.onInterceptTouchEvent().DOWN -> 拦截 -> listView.onTouch() ->
    //DragView.onInterceptTouchEvent().MOVE->DragView.onTouchEvent().MOVE
    //DragView的onTouchEvent缺失了down的处理部分 mViewDragHelper.processTouchEvent(event);也就缺失了down事件
    //事件不完整
    private float mDownY = 0f;

    //3 事件的分发和拦截 需要结合canChildScrollUp使用 canChildScrollUp来自SwipeRefreshLayout的对应方法
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        //菜单打开(展开态-非折叠态)父容器拦截所有事件 listView不应该自己处理事件
        if (isMenuOpen) {
            return true;
        }
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                //让mViewDragHelper获取完整事件 否则mViewDragHelper会丢弃整个事件(只得到move事件不处理)
                mViewDragHelper.processTouchEvent(ev);
                mDownY = ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                float moveY = ev.getY();
                //菜单关闭 (折叠态)并且是向下滑动 并且listView内部不能向下滚动 父容器拦截所有事件 listView不应该自己处理事件
                if (moveY > mDownY && !canChildScrollUp()) {
                    return true;
                }
                break;
        }
        return super.onInterceptTouchEvent(ev);
    }

    /**
     * @return Whether it is possible for the child view of this layout to
     * scroll up. Override this if the child view is a custom view.
     * 判断View是否滚动到了最顶部,还能不能向上滚
     * 这个方法比较老了
     */
    public boolean canChildScrollUpOld() {
        if (android.os.Build.VERSION.SDK_INT < 14) {
            if (frontView instanceof AbsListView) {
                final AbsListView absListView = (AbsListView) frontView;
                return absListView.getChildCount() > 0
                        && (absListView.getFirstVisiblePosition() > 0 || absListView.getChildAt(0)
                        .getTop() < absListView.getPaddingTop());
            } else {
                return ViewCompat.canScrollVertically(frontView, -1) || frontView.getScrollY() > 0;
            }
        } else {
            return frontView.canScrollVertically(-1);
        }
    }

    /**
     * @return Whether it is possible for the child view of this layout to
     * scroll up. Override this if the child view is a custom view.
     * 判断View是否滚动到了最顶部,还能不能向上滚
     * 这个是API29最新的code 与canChildScrollUpOld功能相同
     */
    public boolean canChildScrollUp() {
        return frontView instanceof ListView ? ListViewCompat.canScrollList((ListView) frontView, -1) : frontView.canScrollVertically(-1);
    }
}

4.2Activity部分

public class MainActivity extends AppCompatActivity {

    ListView listView;
    List<String> listStrings = new ArrayList<String>();

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        listView = ((DragView) findViewById(R.id.dragView)).getFrontListView();
        for (int i = 0; i < 50; i++) {
            listStrings.add("String-> " + i);
        }
        listView.setAdapter(new BaseAdapter() {
            @Override
            public int getCount() {
                return listStrings.size();
            }

            @Override
            public Object getItem(int position) {
                return null;
            }

            @Override
            public long getItemId(int position) {
                return position;
            }

            @Override
            public View getView(int position, View convertView, ViewGroup parent) {
                LayoutInflater inflater = LayoutInflater.from(MainActivity.this);
                TextView textView = (TextView) inflater.inflate(R.layout.layout_items, parent, false);
                textView.setText(listStrings.get(position));
                return textView;
            }
        });
    }
}

4.3Activity的布局

<?xml version="1.0" encoding="utf-8"?>
<com.example.chj.draglistview.DragView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/dragView"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="400dp"
        android:background="#ccc">

        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:text="后面" />
    </RelativeLayout>

    <ListView
        android:background="#cfc"
        android:id="@+id/listView"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</com.example.chj.draglistview.DragView>

代码
https://github.com/caihuijian/learn_darren_android.git

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值