最近在做项目的时候偶然间看到QMUI中的QMUIPopup弹出的时候点击按钮不会反复弹出,于是有了点兴趣就去看了一下他的源码,作为学习记录。
你是否和我一样在实现全屏的布局还在使用布局嵌套的方式,来进行全屏的dialog展示呢?接下来我们将要告别这种布局嵌套的方式来进行全屏展示。
弹窗的背景实现,是操作window
来实现的。弹出前修改window透明度
,弹出后将window
的透明度修改回去,这样就不用嵌套布局了。
//1.通过window对象获取属性
WindowManager.LayoutParams attributes=getWindow().getAttributes();
//2.修改属性的透明度 取值范围:0.0f~1.0f
attributes.alpha = 0.4f;
//3.将属性设置给window
getWindow().setAttributes(attributes);
//4.Android10以后需要设置
getWindow().addFlags(WindowManager.LayoutParams.FLAG_DIM_BEHIND);
在setOnDismissListener
中将window
设置回来。
设置以下属性:
//需要使用软键盘
setInputMethodMode(PopupWindow.INPUT_METHOD_NEEDED);
setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_PAN);
//将焦点设置给弹窗,否则弹出软键盘输入内容不会进入弹窗EditText中
setFocusable(true);
设置以下属性:
//设置背景颜色
setBackgroundDrawable(new ColorDrawable(ContextCompat.getColor(context, R.color.transparent)));
setOutsideTouchable(true);
setTouchable(true);
//为所有被分派到弹出窗口的触摸事件设置一个回调(来自源码翻译)
setTouchInterceptor(new View.OnTouchListener() {
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
if(event.getAction()== MotionEvent.ACTION_OUTSIDE){
Log.e("zzz","一些事件发生在窗口外部:操作在外部");
return true;
}
Log.e("zzz","某个事件发生在窗口外部");
return false;
}
}
});
四.QMUI源码讲解
接下来我们就需要去查看PopupWindow中的setTouchInterceptor
函数。
setTouchIntercepor
:是获取弹窗上所有的事件。