Android自定义FloatingActionButton实现

作者: Android师哥 | 来源:发表于2019-01-04 11:19 被阅读2次
    night_rain.png

    自定义View,一是为了满足设计需求,二是开发者进阶的标志之一。随心所欲就是我等奋斗的目标!!!

    效果

    拖动按钮

    实现逻辑

    • 明确需求

      在做任何事情之前都要先明确需求,开发更是重要。我们现在要实现一个可以在屏幕上拖动的控件,并且有自动贴边的效果!

    • 整理思路

      1. 既然要实现控件拖动,那么就离不开onTouchEvent()这个方法,需要监听里面的按下和滑动事件。
      2. 要实现自动贴边,需要监听onTouchEvent()中手指离开屏幕事件。对于贴边的过程,我们用属性动画来解决。
      3. 事件的冲突问题也需要考虑,拖动、点击关系到了事件的拦截。
    • 动手实现

      在需求明确、思路清晰的情况下就要开始动手实现(需要了解自定义View的一些基础API),下面代码中注释写的基本都差不多,很好理解。欢迎指出讨论!!!

    完整代码

    public class AttachButton extends FloatingActionButton {
    
        private float mLastRawX;
        private float mLastRawY;
        private final String TAG = "AttachButton";
        private boolean isDrug = false;
        private int mRootMeasuredWidth;
    
        public AttachButton(Context context) {
            this(context, null);
        }
    
        public AttachButton(Context context, AttributeSet attrs) {
            this(context, attrs, 0);
        }
    
        public AttachButton(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent ev) {
            //当前手指的坐标
            float mRawX = ev.getRawX();
            float mRawY = ev.getRawY();
            switch (ev.getAction()) {
                case MotionEvent.ACTION_DOWN://手指按下
                    isDrug = false;
                    //记录按下的位置
                    mLastRawX = mRawX;
                    mLastRawY = mRawY;
                    break;
                case MotionEvent.ACTION_MOVE://手指滑动
                    ViewGroup mViewGroup = (ViewGroup) getParent();
                    if (mViewGroup != null) {
                        int[] location = new int[2];
                        mViewGroup.getLocationInWindow(location);
                        //获取父布局的高度
                        int mRootMeasuredHeight = mViewGroup.getMeasuredHeight();
                        mRootMeasuredWidth = mViewGroup.getMeasuredWidth();
                        //获取父布局顶点的坐标
                        int mRootTopY = location[1];
                        if (mRawX >= 0 && mRawX <= mRootMeasuredWidth && mRawY >= mRootTopY && mRawY <= (mRootMeasuredHeight + mRootTopY)) {
                            //手指X轴滑动距离
                            float differenceValueX = mRawX - mLastRawX;
                            //手指Y轴滑动距离
                            float differenceValueY = mRawY - mLastRawY;
                            //判断是否为拖动操作
                            if (!isDrug) {
                                if (Math.sqrt(differenceValueX * differenceValueX + differenceValueY * differenceValueY) < 2) {
                                    isDrug = false;
                                } else {
                                    isDrug = true;
                                }
                            }
                            //获取手指按下的距离与控件本身X轴的距离
                            float ownX = getX();
                            //获取手指按下的距离与控件本身Y轴的距离
                            float ownY = getY();
                            //理论中X轴拖动的距离
                            float endX = ownX + differenceValueX;
                            //理论中Y轴拖动的距离
                            float endY = ownY + differenceValueY;
                            //X轴可以拖动的最大距离
                            float maxX = mRootMeasuredWidth - getWidth();
                            //Y轴可以拖动的最大距离
                            float maxY = mRootMeasuredHeight - getHeight();
                            //X轴边界限制
                            endX = endX < 0 ? 0 : endX > maxX ? maxX : endX;
                            //Y轴边界限制
                            endY = endY < 0 ? 0 : endY > maxY ? maxY : endY;
                            //开始移动
                            setX(endX);
                            setY(endY);
                            //记录位置
                            mLastRawX = mRawX;
                            mLastRawY = mRawY;
                        }
                    }
                    break;
                case MotionEvent.ACTION_UP://手指离开
                    float center = mRootMeasuredWidth / 2;
                    //自动贴边
                    if (mLastRawX <= center) {
                        //向左贴边
                        AttachButton.this.animate()
                                .setInterpolator(new BounceInterpolator())
                                .setDuration(500)
                                .x(0)
                                .start();
                    } else {
                        //向右贴边
                        AttachButton.this.animate()
                                .setInterpolator(new BounceInterpolator())
                                .setDuration(500)
                                .x(mRootMeasuredWidth - getWidth())
                                .start();
                    }
                    break;
            }
            //是否拦截事件
            return isDrug ? isDrug : super.onTouchEvent(ev);
        }
    }
    

    相关文章

      网友评论

        本文标题:Android自定义FloatingActionButton实现

        本文链接:https://www.haomeiwen.com/subject/lpjhrqtx.html