悬浮窗

一、原理

1、添加独立的View

我们在APP中想不依赖Activity中的布局添加View时,可以通过WindowManager.addView()的方式,创建一个window,并显示添加的View。

2、Window可分为三类

应用window:一般位于最底层,对应一个Activity;
子window:不能单独存在,需要附属在父window上,如Dialog;
系统window:一般位于最顶层,不会被其他window遮住,如Toast。

二、悬浮窗

1、检查权限
// 检查悬浮窗
Settings.canDrawOverlays(mContext)

// 跳转到悬浮窗设置页面
activity.startActivityForResult(new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                Uri.parse("package:" + activity.getPackageName())), requestCode);

2、添加悬浮窗

此处需要注意的是flag对于悬浮窗的影响:
FLAG_NOT_TOUCH_MODAL: 悬浮窗外部可响应事件
FLAG_NOT_FOCUSABLE : 不处理系统按钮事件
FLAG_LAYOUT_NO_LIMITS: 悬浮窗可以延伸到屏幕外

if (Settings.canDrawOverlays(mContext)) {
    // 获取WindowManager服务
    mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
    mLayoutParams = new WindowManager.LayoutParams();
    // 设置LayoutParam
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        mLayoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
    } else {
        mLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
    }
    // 悬浮窗外部可响应事件
    mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
            // 不处理系统按钮事件
            WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |
            // 悬浮窗可以延伸到屏幕外
            WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS;
    mLayoutParams.format = PixelFormat.RGBA_8888;
    mLayoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
    mLayoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
    mLayoutParams.gravity = Gravity.TOP | Gravity.START;
    mLayoutParams.x = 0;
    mLayoutParams.y = 300;
    // 将悬浮窗控件添加到WindowManager
    mWindowManager.addView(mRootView, mLayoutParams);
}


// 移除悬浮窗
try {
    mWindowManager.removeView(mRootView);
} catch (Exception e) {
    //nothing
}
3、响应点击事件
// 可以给view直接设置
mRootView.setOnClickListener(onClickListener);
4、拖动

首先, 在ACTION_DOWN中记录当前的x,y位置;
然后, 在ACTION_MOVE中判断滑动距离,如果触发了滑动,算出滑动后的x,y位置,通过updateViewLayout进行更新;
最后, 在ACTION_UP中处理贴边,点击等逻辑。

// 拖动可以给view设置时间监听
if (canDrag) {
    mRootView.setOnTouchListener(new EFWindowOnTouchListener());
}

private class EFWindowOnTouchListener implements View.OnTouchListener {
        private boolean isMoved;
        private int oX;
        private int oY;
        private int x;
        private int y;

        @Override
        public boolean onTouch(View view, MotionEvent event) {
            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    // 初始化滑动的标记
                    isMoved = false;
                    oX = (int) event.getRawX();
                    oY = (int) event.getRawY();
                    x = (int) event.getRawX();
                    y = (int) event.getRawY();
                    break;
                case MotionEvent.ACTION_MOVE:
                    int nowX = (int) event.getRawX();
                    int nowY = (int) event.getRawY();
                    int movedX = nowX - x;
                    int movedY = nowY - y;
                    x = nowX;
                    y = nowY;

                    // 如果没有触发滑动,则进行滑动判断
                    if (!isMoved && (Math.abs(oX - x) > mMinScrollDistance || Math.abs(oY - y) > mMinScrollDistance)) {
                        isMoved = true;
                    }

                    // 拖动中不能用贴边计算
                    mLayoutParams.x = getRealX(mLayoutParams.x + movedX, false, false);
                    mLayoutParams.y = getRealY(mLayoutParams.y + movedY, false);
                    mWindowManager.updateViewLayout(view, mLayoutParams);
                    break;
                case MotionEvent.ACTION_UP:
                    if (mIsSticky) {
                        int viewWidth = view.getMeasuredWidth();
                        int screenWidth = ScreenUtils.getScreenWidth(mContext);
                        // 贴边终点只有左、右两种状态
                        if ((mLayoutParams.x + viewWidth / 2) > screenWidth / 2) {
                            if (mOnPositionListener != null) {
                                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
                            }
                            mLayoutParams.x = screenWidth - viewWidth - mXEdgeSize;
                        } else {
                            if (mOnPositionListener != null) {
                                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
                            }
                            mLayoutParams.x = mXEdgeSize;
                        }
                        mWindowManager.updateViewLayout(view, mLayoutParams);
                    }

                    // 处理移动后还会响应点击事件的情况
                    if (isMoved) {
                        isMoved = false;
                        return true;
                    }
                    break;
                default:
                    break;
            }
            return false;
        }
    }
5、横竖屏切换适配

监听到屏幕变化后,由于屏幕的长宽发生了变化,重新计算当前的x,y,并进行更新。

// 添加屏幕变化监听
DisplayManager displayManager = (DisplayManager) mContext.getSystemService(Context.DISPLAY_SERVICE);
displayManager.registerDisplayListener(new DisplayManager.DisplayListener() {
    @Override
    public void onDisplayAdded(int displayId) {
    }
    @Override
    public void onDisplayRemoved(int displayId) {
    }
    @Override
    public void onDisplayChanged(int displayId) {
        // 监听到屏幕发生变化
        if (mLayoutParams != null && mWindowManager != null
                && mRootView != null && mRootView.isAttachedToWindow()) {
            mLayoutParams.x = getRealX(mLayoutParams.x, false, true);
            mLayoutParams.y = getRealY(mLayoutParams.y, false);
            mWindowManager.updateViewLayout(mRootView, mLayoutParams);
        }
    }
}, new Handler(Looper.getMainLooper()));

6、贴边,缩进等处理
private int getRealX(int xPosition, boolean isInit, boolean considerSticky) {
    if (isInit) {
        // 此时view还没有触发测量,主动测量一次
        getRootView().measure(View.MeasureSpec.UNSPECIFIED, View.MeasureSpec.UNSPECIFIED);
    }
    int viewWidth = getRootView().getMeasuredWidth();
    int screenWidth = ScreenUtils.getScreenWidth(mContext);
    int screenHeight = ScreenUtils.getScreenHeight(mContext);
    // 部分手机,如:vivo R9s,横竖屏切换,获取的值不会发生变化,此处做一个适配
    screenWidth = isHorizontalScreen(mContext) ? Math.max(screenWidth, screenHeight) : Math.min(screenWidth, screenHeight);
    // 贴边初始只有左和右,没有中间态
    if (considerSticky && mIsSticky) {
        if ((xPosition + viewWidth / 2) > screenWidth / 2) {// 处理缩进
            if (mOnPositionListener != null) {
                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
            }
            // 贴右边显示,真实坐标为(屏幕宽度 - view的宽度 - x边距)
            return screenWidth - viewWidth - mXEdgeSize;
        } else {
            if (mOnPositionListener != null) {
                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
            }
            // 贴左边显示,真实坐标为(左起点 + x边距),左边的起点为0。
            return mXEdgeSize;
        }
    } else {
        if (xPosition > (screenWidth - viewWidth - mXEdgeSize)) {
            if (mOnPositionListener != null) {
                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
            }
            // 当前位置已经超过允许的右边距,返回最右边距
            return screenWidth - viewWidth - mXEdgeSize;
        } else if (xPosition < mXEdgeSize) {
            if (mOnPositionListener != null) {
                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
            }
            // 当前位置小于允许的左边距,返回最左边距
            return mXEdgeSize;
        } else {
            if (mOnPositionListener != null) {
                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_MIDDLE);
            }
            // 由于不吸边,所以展示真实位置
            return xPosition;
        }
    }
}


private int getRealY(int yPosition, boolean isInit) {
    // 此时view还没有触发测量,主动测量一次
    if (isInit) {
        getRootView().measure(View.MeasureSpec.UNSPECIFIED, View.MeasureSpec.UNSPECIFIED);
    }
    int viewHeight = getRootView().getMeasuredHeight();
    int screenWidth = ScreenUtils.getScreenWidth(mContext);
    int screenHeight = ScreenUtils.getScreenHeight(mContext);
    // 部分手机,如:vivo R9s,横竖屏切换,获取的值不会发生变化,此处做一个适配
    screenHeight = isHorizontalScreen(mContext) ? Math.min(screenWidth, screenHeight) : Math.max(screenWidth, screenHeight);
    if (yPosition > (screenHeight - viewHeight - mYBottomEdgeSize)) {
        // 当前位置已经超过允许的下边距,返回最下边距
        return screenHeight - viewHeight - mYBottomEdgeSize;
    } else if (yPosition < mYTopEdgeSize) {
        // 当前位置小于允许的上边距,返回最上边距
        return mYTopEdgeSize;
    } else {
        // 展示真实位置
        return yPosition;
    }
}

private boolean isHorizontalScreen(Context context) {
    int angle = ((WindowManager) context.getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay().getRotation();
    //屏幕旋转90°或270°,表示横屏
    return angle == Surface.ROTATION_90 || angle == Surface.ROTATION_270;
}

三、注意

部分场景需要点击回到当前的Activity,可以参考如下代码:

// 方法一:跳转到app展示的上一个页面
public void onClick(View v) {
    Intent intent = BaseApplication.getGlobalContext().getPackageManager().getLaunchIntentForPackage(BaseApplication.getGlobalContext().getPackageName());
    BaseApplication.getGlobalContext().startActivity(intent);
}

// 方法二:本地缓存页面,判断登录状态跳转不同的页面
public void onClick(View v) {
    // 注意,这里的跳转没有参数传递,在一些activity的onNewIntent中会对intent进行再次解析
    // 因此,需要判断是否是从悬浮窗跳过去的,进行拦截,防止出现异常
    // 也可以将正确的参数传递过去,成本较高,用作备选项
    // FLAG_ACTIVITY_SINGLE_TOP,如果当前activity已经在栈顶,不再创建,直接跳转
    // FLAG_ACTIVITY_REORDER_TO_FRONT,如果activity已经创建了,如:ABCD,跳转到B,不会重新创建,且变为ACDB
    if (ACCOUNT_CONTROLLER.hasLogin()) {
        if (mCurrentActivity != null) {
            // 正常情况都不是null,除非长时间被系统回收了
            Intent targetIntent = new Intent(BaseApplication.getGlobalContext(), mCurrentActivity.getClass());
            targetIntent.putExtra(SpName.INTENT_FROM, FROM_EFWINDOW);
            targetIntent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_CLEAR_TOP);
            mCurrentActivity.startActivity(targetIntent);
        } else {
            // 跳转首页,然后通过恢复接口进行再次跳转
            Intent targetIntent = new Intent(BaseApplication.getGlobalContext(), HomeActivity.class);
            targetIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_CLEAR_TOP);
            targetIntent.putExtra(SpName.INTENT_FROM, FROM_EFWINDOW);
            BaseApplication.getGlobalContext().startActivity(targetIntent);
        }
    } else {
        // 登录信息失效了
        Intent targetIntent = new Intent(BaseApplication.getGlobalContext(), LoginActivity.class);
        targetIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_SINGLE_TOP | Intent.FLAG_ACTIVITY_CLEAR_TOP);
        targetIntent.putExtra(SpName.INTENT_FROM, FROM_EFWINDOW);
        BaseApplication.getGlobalContext().startActivity(targetIntent);
        ToastUtil.showMessage("登录信息失效,请重新登录");
    }
}

四、悬浮窗代码

//try {
//    if (mEfWindow == null) {
//        View view = LayoutInflater.from(BaseApplication.getGlobalContext()).inflate(R.layout.layout_edj_floating_window, null);
//        mEfWindow = new EFWindow.Builder(BaseApplication.getGlobalContext())
//                .view(view)
//                .yPosition(ScreenUtils.getScreenHeight(BaseApplication.getGlobalContext()) / 3)
//                .isSticky(true)
//                .canDrag(true)
//                .xEdgeSize(-UIUtils.dp2px(BaseApplication.getGlobalContext(), 20))
//                .onClickListener(this)
//                .build();
//    }
//    mEfWindow.show();
//} catch (Exception e) {
//    e.printStackTrace();
//}
public class EFWindow {
    private final int mMinScrollDistance;
    private WindowManager mWindowManager;
    private WindowManager.LayoutParams mLayoutParams;
    private final View mRootView;
    private boolean mIsShow;
    private boolean mIsSticky;
    private int mXEdgeSize;
    private int mYTopEdgeSize;
    private int mYBottomEdgeSize;
    private OnPositionListener mOnPositionListener;
    private final Context mContext;

    @RequiresApi(api = Build.VERSION_CODES.M)
    private EFWindow(Builder builder) {
        // 赋值
        mContext = builder.mContext;
        mMinScrollDistance = ViewConfiguration.get(mContext).getScaledTouchSlop();
        mIsSticky = builder.mIsSticky;
        mXEdgeSize = builder.mXEdgeSize;
        mYTopEdgeSize = builder.mYTopEdgeSize;
        mYBottomEdgeSize = builder.mYBottomEdgeSize;
        mOnPositionListener = builder.mOnPositionListener;
        mRootView = builder.mRootView;
        boolean canDrag = builder.mCanDrag;
        int xPosition = builder.mXPosition;
        int yPosition = builder.mYPosition;
        View.OnClickListener onClickListener = builder.mOnClickListener;

        // 添加屏幕变化监听
        DisplayManager displayManager = (DisplayManager) mContext.getSystemService(Context.DISPLAY_SERVICE);
        displayManager.registerDisplayListener(new DisplayManager.DisplayListener() {
            @Override
            public void onDisplayAdded(int displayId) {

            }

            @Override
            public void onDisplayRemoved(int displayId) {

            }

            @Override
            public void onDisplayChanged(int displayId) {
                // 监听到屏幕发生变化
                if (mLayoutParams != null && mWindowManager != null
                        && mRootView != null && mRootView.isAttachedToWindow()) {
                    mLayoutParams.x = getRealX(mLayoutParams.x, false, true);
                    mLayoutParams.y = getRealY(mLayoutParams.y, false);
                    mWindowManager.updateViewLayout(mRootView, mLayoutParams);
                }
            }
        }, new Handler(Looper.getMainLooper()));

        if (Settings.canDrawOverlays(mContext)) {
            // 获取WindowManager服务
            mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
            if (canDrag) {
                mRootView.setOnTouchListener(new EFWindowOnTouchListener());
            }
            mRootView.setOnClickListener(onClickListener);

            mLayoutParams = new WindowManager.LayoutParams();
            // 设置LayoutParam
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                mLayoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
            } else {
                mLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
            }
            // 悬浮窗外部可响应事件
            mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
                    // 不处理系统按钮事件
                    WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE |
                    // 悬浮窗可以延伸到屏幕外
                    WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS;
            mLayoutParams.format = PixelFormat.RGBA_8888;
            mLayoutParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
            mLayoutParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
            mLayoutParams.gravity = Gravity.TOP | Gravity.START;
            mLayoutParams.x = getRealX(xPosition, true, true);
            mLayoutParams.y = getRealY(yPosition, true);
        }
    }

    private int getRealX(int xPosition, boolean isInit, boolean considerSticky) {
        if (isInit) {
            // 此时view还没有触发测量,主动测量一次
            getRootView().measure(View.MeasureSpec.UNSPECIFIED, View.MeasureSpec.UNSPECIFIED);
        }
        int viewWidth = getRootView().getMeasuredWidth();
        int screenWidth = ScreenUtils.getScreenWidth(mContext);
        int screenHeight = ScreenUtils.getScreenHeight(mContext);

        // 部分手机,如:vivo R9s,横竖屏切换,获取的值不会发生变化,此处做一个适配
        screenWidth = isHorizontalScreen(mContext) ? Math.max(screenWidth, screenHeight) : Math.min(screenWidth, screenHeight);

        // 贴边初始只有左和右,没有中间态
        if (considerSticky && mIsSticky) {
            if ((xPosition + viewWidth / 2) > screenWidth / 2) {// 处理缩进
                if (mOnPositionListener != null) {
                    mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
                }
                // 贴右边显示,真实坐标为(屏幕宽度 - view的宽度 - x边距)
                return screenWidth - viewWidth - mXEdgeSize;
            } else {
                if (mOnPositionListener != null) {
                    mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
                }
                // 贴左边显示,真实坐标为(左起点 + x边距),左边的起点为0。
                return mXEdgeSize;
            }
        } else {
            if (xPosition > (screenWidth - viewWidth - mXEdgeSize)) {
                if (mOnPositionListener != null) {
                    mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
                }
                // 当前位置已经超过允许的右边距,返回最右边距
                return screenWidth - viewWidth - mXEdgeSize;
            } else if (xPosition < mXEdgeSize) {
                if (mOnPositionListener != null) {
                    mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
                }
                // 当前位置小于允许的左边距,返回最左边距
                return mXEdgeSize;
            } else {
                if (mOnPositionListener != null) {
                    mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_MIDDLE);
                }
                // 由于不吸边,所以展示真实位置
                return xPosition;
            }
        }
    }

    private int getRealY(int yPosition, boolean isInit) {
        // 此时view还没有触发测量,主动测量一次
        if (isInit) {
            getRootView().measure(View.MeasureSpec.UNSPECIFIED, View.MeasureSpec.UNSPECIFIED);
        }
        int viewHeight = getRootView().getMeasuredHeight();
        int screenWidth = ScreenUtils.getScreenWidth(mContext);
        int screenHeight = ScreenUtils.getScreenHeight(mContext);

        // 部分手机,如:vivo R9s,横竖屏切换,获取的值不会发生变化,此处做一个适配
        screenHeight = isHorizontalScreen(mContext) ? Math.min(screenWidth, screenHeight) : Math.max(screenWidth, screenHeight);

        if (yPosition > (screenHeight - viewHeight - mYBottomEdgeSize)) {
            // 当前位置已经超过允许的下边距,返回最下边距
            return screenHeight - viewHeight - mYBottomEdgeSize;
        } else if (yPosition < mYTopEdgeSize) {
            // 当前位置小于允许的上边距,返回最上边距
            return mYTopEdgeSize;
        } else {
            // 展示真实位置
            return yPosition;
        }
    }

    public View getRootView() {
        return mRootView;
    }

    public boolean isShow() {
        return mIsShow;
    }

    public void show() {
        mIsShow = true;

        // 先尝试移除
        try {
            mLayoutParams.x = getRealX(mLayoutParams.x, false, true);
            mLayoutParams.y = getRealY(mLayoutParams.y, false);
            mWindowManager.removeView(mRootView);
        } catch (Exception e) {
            //nothing
        }
        // 将悬浮窗控件添加到WindowManager
        mWindowManager.addView(mRootView, mLayoutParams);
    }

    public void dismiss() {
        // 尝试移除
        try {
            mWindowManager.removeView(mRootView);
        } catch (Exception e) {
            //nothing
        }
        mIsShow = false;
    }


    private class EFWindowOnTouchListener implements View.OnTouchListener {
        private boolean isMoved;
        private int oX;
        private int oY;
        private int x;
        private int y;

        @Override
        public boolean onTouch(View view, MotionEvent event) {
            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    // 初始化滑动的标记
                    isMoved = false;
                    oX = (int) event.getRawX();
                    oY = (int) event.getRawY();
                    x = (int) event.getRawX();
                    y = (int) event.getRawY();
                    break;
                case MotionEvent.ACTION_MOVE:
                    int nowX = (int) event.getRawX();
                    int nowY = (int) event.getRawY();
                    int movedX = nowX - x;
                    int movedY = nowY - y;
                    x = nowX;
                    y = nowY;

                    // 如果没有触发滑动,则进行滑动判断
                    if (!isMoved && (Math.abs(oX - x) > mMinScrollDistance || Math.abs(oY - y) > mMinScrollDistance)) {
                        isMoved = true;
                    }

                    // 拖动中不能用贴边计算
                    mLayoutParams.x = getRealX(mLayoutParams.x + movedX, false, false);
                    mLayoutParams.y = getRealY(mLayoutParams.y + movedY, false);
                    mWindowManager.updateViewLayout(view, mLayoutParams);
                    break;
                case MotionEvent.ACTION_UP:
                    if (mIsSticky) {
                        int viewWidth = view.getMeasuredWidth();
                        int screenWidth = ScreenUtils.getScreenWidth(mContext);
                        // 贴边终点只有左、右两种状态
                        if ((mLayoutParams.x + viewWidth / 2) > screenWidth / 2) {
                            if (mOnPositionListener != null) {
                                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_RIGHT);
                            }
                            mLayoutParams.x = screenWidth - viewWidth - mXEdgeSize;
                        } else {
                            if (mOnPositionListener != null) {
                                mOnPositionListener.onPositionChanged(OnPositionListener.POSITION_LEFT);
                            }
                            mLayoutParams.x = mXEdgeSize;
                        }
                        mWindowManager.updateViewLayout(view, mLayoutParams);
                    }

                    // 处理移动后还会响应点击事件的情况
                    if (isMoved) {
                        isMoved = false;
                        return true;
                    }
                    break;
                default:
                    break;
            }
            return false;
        }
    }

    public static class Builder {
        private final Context mContext;
        private View mRootView;
        private int mXPosition;
        private int mYPosition;
        private boolean mCanDrag;
        private boolean mIsSticky;
        private int mXEdgeSize;
        private int mYTopEdgeSize;
        private int mYBottomEdgeSize;
        private View.OnClickListener mOnClickListener;
        private OnPositionListener mOnPositionListener;

        public Builder(Context context) {
            mContext = context;
        }

        public Builder onPositionListener(OnPositionListener onPositionListener) {
            mOnPositionListener = onPositionListener;
            return this;
        }

        public Builder onClickListener(View.OnClickListener onClickListener) {
            mOnClickListener = onClickListener;
            return this;
        }

        public Builder xPosition(int xPosition) {
            mXPosition = xPosition;
            return this;
        }

        public Builder yPosition(int yPosition) {
            mYPosition = yPosition;
            return this;
        }

        public Builder view(View view) {
            mRootView = view;
            return this;
        }

        public Builder canDrag(boolean canDrag) {
            mCanDrag = canDrag;
            return this;
        }

        public Builder isSticky(boolean isSticky) {
            mIsSticky = isSticky;
            return this;
        }

        public Builder xEdgeSize(int xEdgeSize) {
            mXEdgeSize = xEdgeSize;
            return this;
        }

        public Builder yTopEdgeSize(int yTopEdgeSize) {
            mYTopEdgeSize = yTopEdgeSize;
            return this;
        }

        public Builder yBottomEdgeSize(int yBottomEdgeSize) {
            mYBottomEdgeSize = yBottomEdgeSize;
            return this;
        }

        public Builder view(int viewRes) throws Exception {
            if (mContext == null) {
                throw new Exception("EFWindow:请传入正确的context.");
            }
            mRootView = LayoutInflater.from(mContext).inflate(viewRes, null);
            return this;
        }

        @RequiresApi(api = Build.VERSION_CODES.M)
        public EFWindow build() throws Exception {
            if (mContext == null) {
                throw new Exception("EFWindow:请传入正确的context.");
            }
            if (mRootView == null) {
                throw new Exception("EFWindow:请先设置Window布局.");
            }
            return new EFWindow(this);
        }

    }

    public interface OnPositionListener {
        int POSITION_LEFT = 0;
        int POSITION_MIDDLE = 1;
        int POSITION_RIGHT = 2;

        void onPositionChanged(int positionStatus);
    }

    private boolean isHorizontalScreen(Context context) {
        int angle = ((WindowManager) context.getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay().getRotation();
        //屏幕旋转90°或270°,表示横屏
        return angle == Surface.ROTATION_90 || angle == Surface.ROTATION_270;
    }
}

你可能感兴趣的:(悬浮窗)