一、点击外部取消显示
一般来说,在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; }