一、点击外部取消显示
一般来说,在android中用到的都是设置dialog点击外部取消dialog的显示,这个直接设置setCanceledOnTouchOutside方法即可。
public void setCanceledOnTouchOutside(boolean cancel) {
if (cancel && !mCancelable) {
mCancelable = true;
}
//下面是调用window对象设置
mWindow.setCloseOnTouchOutside(cancel);
}
点击外部,最终还是会进入onTouchEvent事件的,
public boolean onTouchEvent(MotionEvent event) {
//mWindow.shouldCloseOnTouch(mContext, event)返回值就是上面的设置值,true会进入
//cancel方法。
if (mCancelable && mShowing && mWindow.shouldCloseOnTouch(mContext, event)) {
cancel();
return true;
}
return false;
}
cancel方法会调用dismissDialog方法,然后通过windowmanager移除dialog的顶层decor视图(view)就可以了,源码还是很简单的,想看的可以自己跟一下。
二、点击外部自己监听事件
但是有时候我们不仅仅是需要设置点击外部取消,而是想做一些其他的动作,比如点击外部跳转到其他Activity等,在网上找到一种思路。首先在View创建之前设置两个Flag,一个设置窗口为非模式的,这样除窗口外的内容就可以获得touch事件,然后设置窗口外部touch事件发生时的通知。最后重写onTouchEvent,监听窗口外的Touch事件。这样就可以在监听方法中自定义窗口外点击事件的响应,是否关闭窗口或者其他操作。
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// Make us non-modal, so that others can receive touch events.
getWindow().setFlags(LayoutParams.FLAG_NOT_TOUCH_MODAL, LayoutParams.FLAG_NOT_TOUCH_MODAL);
// ...but notify us that it happened.
getWindow().setFlags(LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH, LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH);
// Note that flag changes must happen *before* the content view is set.
setContentView(R.layout.my_dialog_view);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
// If we've received a touch notification that the user has touched
// outside the app, finish the activity.
if (MotionEvent.ACTION_OUTSIDE == event.getAction()) {
finish();
return true;
}
// Delegate everything else to Activity.
return super.onTouchEvent(event);
}
这种方式需要dialog下面的Activity没有点击事件,不过这种可能性很小,因此点击的时候事件会传递到dialog下面的Activity的view,因此实用性不强。
三、监听点击位置设置事件
这种方式我用着是最好的,实用性比较强,就是根据点击的位置判断是不是在dialog的外部。
@Override
public boolean onTouchEvent(MotionEvent event) {
if (event.getAction() == MotionEvent.ACTION_DOWN) {
if (!(event.getX() >= -10 && event.getY() >= -10)
|| event.getX() >= calendarLayout.getWidth() + 10
|| event.getY() >= calendarLayout.getHeight() + 20) {//如果点击位置在当前View外部则销毁当前视图,其中10与20为微调距离
finish();
}
}
return true;
}
Android中Dialog设置外部点击事件 |