VerticalDragLayout

本文详细介绍了VerticalDragLayout类的设计与实现,包括ViewDragHelper的使用、滑动监听与事件处理,以及如何实现垂直方向上的流畅拖动效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

package com.eoo.eoodemo.taobao;

import android.content.Context;
import android.support.v4.view.ViewCompat;
import android.support.v4.widget.ViewDragHelper;
import android.util.AttributeSet;
import android.util.Log;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;
import android.widget.AbsListView;
import android.widget.FrameLayout;

/**
 * Created by llchen.Andy on 2015/12/18.
 */
public class VerticalDragLayout extends FrameLayout {

    private ViewDragHelper mDragHelper;
    private GestureDetector mGestureDec;

    public VerticalDragLayout(Context context) {
        super(context);
        init();
    }

    public VerticalDragLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    public VerticalDragLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        mDragHelper = ViewDragHelper.create(this, new ViewDragHelper.Callback() {

            @Override
            public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
                super.onViewPositionChanged(changedView, left, top, dx, dy);
                performViewPositionChanged(changedView, dy);
            }

            @Override
            public void onViewReleased(View releasedChild, float xvel, float yvel) {
                super.onViewReleased(releasedChild, xvel, yvel);

                performViewReleased(releasedChild, yvel);
            }

            @Override
            public boolean tryCaptureView(View child, int pointerId) {
                return true;
            }

            @Override
            public int clampViewPositionVertical(View child, int top, int dy) {
                int finalTop = top;
                if (child == getChildAt(0)) {
                    finalTop = Math.min(0, top);
                } else {
                    finalTop = Math.max(0, top);
                }

                return (int) (child.getTop() + (finalTop - child.getTop()) * 0.5f);
            }
        });

        mGestureDec = new GestureDetector(getContext(), new GestureDetector.SimpleOnGestureListener() {
            @Override
            public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {

                if (getChildAt(0).getBottom() > 0) {
                    if (canChildScrollDown(getChildAt(0))) {
                        return false;
                    }
                } else if (canChildScrollDown(getChildAt(1)) && distanceY > 0) {
                    return false;
                } else if (canChildScrollUp(getChildAt(1)) && distanceY < 0) {
                    return false;
                }

                return Math.abs(distanceY) > Math.abs(distanceX);
            }
        });
    }

    private void performViewPositionChanged(View changedView, int dy) {

        View targetView = null;
        if (changedView == getChildAt(1)) {
            targetView = getChildAt(0);
        } else {
            targetView = getChildAt(1);
        }

        targetView.offsetTopAndBottom(dy);

        invalidate();
    }

    private void performViewReleased(View changedView, float yvel) {


        if (changedView == getChildAt(0)) {

            log("Top: " + changedView.getTop());

            if (yvel < -1200 || changedView.getTop() < -100) {
                mDragHelper.smoothSlideViewTo(changedView, changedView.getLeft(), -changedView.getMeasuredHeight());
            } else {
                mDragHelper.smoothSlideViewTo(changedView, changedView.getLeft(), 0);
            }
        } else if (changedView == getChildAt(1)) {
            if (yvel > 1200 || changedView.getTop() > 100) {
                mDragHelper.smoothSlideViewTo(changedView, changedView.getLeft(), changedView.getMeasuredHeight());
            } else {
                mDragHelper.smoothSlideViewTo(changedView, changedView.getLeft(), 0);
            }
        }

        ViewCompat.postInvalidateOnAnimation(this);
    }

    @Override
    public void computeScroll() {
        super.computeScroll();
        if (mDragHelper.continueSettling(true)) {
            ViewCompat.postInvalidateOnAnimation(this);
        }
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
//        super.onLayout(changed, left, top, right, bottom);

        int topOffset = top;
        for (int i = 0; i < getChildCount(); i++) {
            View childView = getChildAt(i);
            int height = childView.getMeasuredHeight();
            childView.layout(left, topOffset, right, topOffset + height);
            topOffset += childView.getMeasuredHeight();
        }
    }


    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {

//        if (getChildAt(0).getBottom() > 0 && getChildAt(0).getTop() < 0) {
//            // view粘到顶部或底部,正在动画中的时候,不处理touch事件
//            return false;
//        }


        boolean shouldIntercept = mDragHelper.shouldInterceptTouchEvent(ev);

        boolean isMoved = mGestureDec.onTouchEvent(ev);


//        log("shouldIntercept: " + shouldIntercept);
//        log("isMoved: " + isMoved);


        if (ev.getAction() == MotionEvent.ACTION_DOWN) {
            // action_down时就让mDragHelper开始工作,否则有时候导致异常 他大爷的
            mDragHelper.processTouchEvent(ev);
        }

        return isMoved & shouldIntercept;
    }

    protected boolean canChildScrollDown(View mTargetView) {
        if (android.os.Build.VERSION.SDK_INT < 14) {
            if (mTargetView instanceof AbsListView) {
                final AbsListView absListView = (AbsListView) mTargetView;
                return absListView.getChildCount() > 0
                        && (absListView.getLastVisiblePosition() < absListView.getChildCount() - 1
                        || absListView.getChildAt(absListView.getChildCount() - 1).getBottom() > absListView.getPaddingBottom());
            } else {
                return ViewCompat.canScrollVertically(mTargetView, 1) || mTargetView.getScrollY() < 0;
            }
        } else {
            return ViewCompat.canScrollVertically(mTargetView, 1);
        }
    }

    public boolean canChildScrollUp(View mTargetView) {
        if (android.os.Build.VERSION.SDK_INT < 14) {
            if (mTargetView instanceof AbsListView) {
                final AbsListView absListView = (AbsListView) mTargetView;
                return absListView.getChildCount() > 0
                        && (absListView.getFirstVisiblePosition() > 0 || absListView.getChildAt(0)
                        .getTop() < absListView.getPaddingTop());
            } else {
                return ViewCompat.canScrollVertically(mTargetView, -1) || mTargetView.getScrollY() > 0;
            }
        } else {
            return ViewCompat.canScrollVertically(mTargetView, -1);
        }
    }

    @Override
    public void requestDisallowInterceptTouchEvent(boolean disallowIntercept) {
        super.requestDisallowInterceptTouchEvent(disallowIntercept);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        mDragHelper.processTouchEvent(event);
        return true;
    }

    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();


    }

    private void log(String msg) {
        Log.i("TEST", msg);
    }
}
前端工程化实战:提升开发效率与质量 前端工程化是现代Web开发中不可或缺的环节,它通过工具化、流程化和规范化的手段,显著提升开发效率、代码质量和团队协作能力。本文将系统介绍前端工程化的核心实践,帮助开发者构建更健壮、可维护的前端项目。 -- 前端工程化概述 1.1 什么是前端工程化? 前端工程化是将软件开发中的工程化思想应用于前端领域,实现从“手工作坊”到“工业化生产”的转变。其核心目标包括: 效率:通过自动化工具减少重复劳动 质量:通过规范化和静态检查保障代码质量 协作:统一开发规范,降低团队协作成本 可维护性:通过模块化、组件化提升长期维护性 1.2 为什么需要工程化? 随着前端技术的发展,工程化成为必然选择: 项目复杂度上升:SPA应用、组件化开发、多端适配等场景需要系统化管理 团队协作需求:统一的代码风格、Git工作流和依赖管理不可或缺 性能优化:自动化构建工具可以实现代码压缩、Tree Shaking等优化 -- 工程化核心工具链 2.1 包管理与依赖控制 现代前端项目依赖管理的关键工具: package.json关键配置示例: 2.2 构建工具 Webpack基础配置: Vite的优势: 基于原生ESM的即时服务启动 闪电般的HMR(热模块替换) 开箱即用的TypeScript支持 2.3 代码质量保障 Git提交拦截配置(.husky/pre-commit): lint-staged配置示例: -- 模块化与组件化开发 3.1 模块化方案演进 Tree Shaking原理: 基于ESM的静态分析 通过sideEffects: false标记无副作用模块 3.2 组件化实践 原子
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值