美文网首页
自定义可拖动的LinearLayout实现悬浮窗效果

自定义可拖动的LinearLayout实现悬浮窗效果

作者: 何婧婧啊 | 来源:发表于2018-10-17 14:05 被阅读0次

之前遇到一个需求,实现一个悬浮窗效果:一个ViewGroup里面包含若干个子控件,ViewGroup有点击事件,然后整体可以拖动。

当时在网上看了几篇博客,基本上都是只有一个View控件的悬浮窗,有的ViewGroup实现的,又会有一些Bug,主要是Touch事件的冲突问题。而且在部分手机、系统上又会有问题(被魅族手机坑惨了,魅族真的与众不同!),因此,只得自己研究研究。

实现思路:重写onInterceptTouchEvent 方法,在 MotionEvent.ACTION_DOWN 手指按下的时候记录下初始坐标,然后在MotionEvent.ACTION_UP 手指离开屏幕的时候记录下结束坐标,通过计算出坐标差值来判断是否移动,如果没有移动则执行点击事件处理。

LinearLayout

/**
 * 自定义可拖动的LinearLayout
 * Created by MS on 2018/6/26.
 */
public class DragViewGroup extends LinearLayout {
    private int lastX, lastY, screenWidth, screenHeight;
    private int downX, downY;//手指落下时候的位置
    private int upX, upY;//手指抬起时候的位置
    public int distanceX, distanceY;//移动距离

    public DragViewGroup(Context context) {
        this(context, null);
    }

    public DragViewGroup(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public DragViewGroup(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        DisplayMetrics dm = getResources().getDisplayMetrics();
        screenWidth = dm.widthPixels;
    }

    //定位
    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        super.onLayout(changed, l, t, r, b);
        //可以在这里确定这个viewGroup的:宽 = r-l.高 = b - t
    }

    public void setScreenHeight(int height) {
        this.screenHeight = height;
    }

    //拦截touch事件
    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        int action = ev.getAction();
        switch (action) {
            case MotionEvent.ACTION_DOWN:
                lastX = (int) ev.getRawX();//设定移动的初始位置相对位置
                lastY = (int) ev.getRawY();
                downX = lastX;
                downY = lastY;
                Log.i("sdfasf", "lastX:" + lastX + "");
                Log.i("sdfasf", "lastY:" + lastY + "");
                break;
            case MotionEvent.ACTION_MOVE://移动
                //event.getRawX()事件点距离屏幕左上角的距离
                int dx = (int) ev.getRawX() - lastX;
                int dy = (int) ev.getRawY() - lastY;
                int left = this.getLeft() + dx;
                int top = this.getTop() + dy;
                int right = this.getRight() + dx;
                int bottom = this.getBottom() + dy;
                if (left < 0) { //最左边
                    left = 0;
                    right = left + this.getWidth();
                }
                if (right > screenWidth) { //最右边
                    right = screenWidth;
                    left = right - this.getWidth();
                }
                if (top < 0) {  //最上边
                    top = 0;
                    bottom = top + this.getHeight();
                }
                if (bottom > screenHeight) {//最下边
                    bottom = screenHeight;
                    top = bottom - this.getHeight();
                }
                this.layout(left, top, right, bottom);//设置控件的新位置
                lastX = (int) ev.getRawX();//再次将滑动其实位置定位
                lastY = (int) ev.getRawY();
                break;
            case MotionEvent.ACTION_UP:
                lastX = (int) ev.getRawX();//设定移动的初始位置相对位置
                lastY = (int) ev.getRawY();
                upX = lastX;
                upY = lastY;
                Log.i("sdfasf", "lastX:" + lastX + "");
                Log.i("sdfasf", "lastY:" + lastY + "");
                distanceX = upX - downX;//记录其移动的距离
                distanceY = upY - downY;
                // 每次移动都要设置其layout,不然移动的view会回到原来的位置
                RelativeLayout.LayoutParams lpFeedback = new RelativeLayout.LayoutParams(
                        RelativeLayout.LayoutParams.WRAP_CONTENT, RelativeLayout.LayoutParams.WRAP_CONTENT);
                lpFeedback.leftMargin = this.getLeft();
                lpFeedback.topMargin = this.getTop();
                lpFeedback.setMargins(this.getLeft(), this.getTop(), 0, 0);
                this.setLayoutParams(lpFeedback);
                break;
            default:
                break;
        }
        return super.onInterceptTouchEvent(ev);
    }
}  

设置监听调用

            case R.id.rel_content://悬浮窗内容点击
                int distanceX = mDragViewGroup.distanceX;
                int distanceY = mDragViewGroup.distanceY;
                if (distanceX==0&&distanceY==0){
                                      //点击事件处理   表示没有移动
                    }
                }
                break;

功能简单,只是感于被魅族手机坑了很久,因此记下。如有不对的地方请大神指教

相关文章

网友评论

      本文标题:自定义可拖动的LinearLayout实现悬浮窗效果

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