170

我已经为我的应用程序实现了一个自定义对话框。我想实现当用户在对话框外单击时,对话框将被关闭。我该怎么做?

4

14 回答 14

378

如果您在对话框之外触摸,您可以使用dialog.setCanceledOnTouchOutside(true);which 将关闭对话框。

就像是,

  Dialog dialog = new Dialog(context)
  dialog.setCanceledOnTouchOutside(true);

或者,如果您的 Dialog 在非模型中,那么,

1 -FLAG_NOT_TOUCH_MODAL为对话框的窗口属性设置标志

Window window = this.getWindow();
window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL,
WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);

2 - 将另一个标志添加到 Windows 属性,FLAG_WATCH_OUTSIDE_TOUCH- 这个标志用于对话框在其可见区域之外接收触摸事件。

3 - 覆盖onTouchEvent()对话框并检查操作类型。如果操作类型为“ MotionEvent.ACTION_OUTSIDE”,则表示用户正在对话区域之外进行交互。因此,在这种情况下,您可以关闭对话或决定要执行的操作。查看普通版?

public boolean onTouchEvent(MotionEvent event)  
{  

       if(event.getAction() == MotionEvent.ACTION_OUTSIDE){  
        System.out.println("TOuch outside the dialog ******************** ");  
               this.dismiss();  
       }  
       return false;  
}  

有关更多信息,请查看如何根据接触点关闭自定义对话框?以及 在对话框区域外触摸时如何关闭非模态对话框

于 2011-12-05T10:47:02.060 回答
21

只需使用

dialog.setCanceledOnTouchOutside(true);
于 2013-06-18T11:19:21.703 回答
17

您可以使用 onTouchEvent 的这个实现。它可以防止在活动下对触摸事件做出反应(如 howettl 所述)。

@Override
public boolean onTouchEvent ( MotionEvent event ) {
  // I only care if the event is an UP action
  if ( event.getAction () == MotionEvent.ACTION_UP ) {
    // create a rect for storing the window rect
    Rect r = new Rect ( 0, 0, 0, 0 );
    // retrieve the windows rect
    this.getWindow ().getDecorView ().getHitRect ( r );
    // check if the event position is inside the window rect
    boolean intersects = r.contains ( (int) event.getX (), (int) event.getY () );
    // if the event is not inside then we can close the activity
    if ( !intersects ) {
      // close the activity
      this.finish ();
      // notify that we consumed this event
      return true;
    }
  }
  // let the system handle the event
  return super.onTouchEvent ( event );
}

来源:http ://blog.twimager.com/2010/08/closing-activity-by-touching-outside.html

于 2012-03-30T22:08:09.843 回答
9

或者,如果您正在使用样式 xml 中定义的主题自定义对话框,请将这一行放在您的主题中:

<item name="android:windowCloseOnTouchOutside">true</item>
于 2013-07-30T09:18:00.760 回答
8
dialog.setCanceledOnTouchOutside(true); 

关闭外部触摸对话框。

如果您不想关闭外部触摸,请使用以下代码:

dialog.setCanceledOnTouchOutside(false);
于 2014-07-13T15:04:19.083 回答
7

此方法应完全避免灰色区域以下的活动检索点击事件。

如果有,请删除此行:

window.setFlags(WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL, WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL);

把它放在你创建的活动上

getWindow().setFlags(LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH, LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH);

然后用这个覆盖触摸事件

@Override
public boolean onTouchEvent(MotionEvent ev)
{
    if(MotionEvent.ACTION_DOWN == ev.getAction())
    {
        Rect dialogBounds = new Rect();
        getWindow().getDecorView().getHitRect(dialogBounds);
        if (!dialogBounds.contains((int) ev.getX(), (int) ev.getY())) {
            // You have clicked the grey area
            displayYourDialog();
            return false; // stop activity closing
        }
    }

    // Touch events inside are fine.
    return super.onTouchEvent(ev);
}
于 2014-06-26T16:06:46.453 回答
6

你可以试试这个:-

AlterDialog alterdialog;
alertDialog.setCanceledOnTouchOutside(true);

或者

alertDialog.setCancelable(true);

如果你有一个AlterDialog.BuilderThen 你可以试试这个: -

alertDialogBuilder.setCancelable(true);
于 2016-12-27T10:31:25.877 回答
5

此代码用于在单击对话框时使用 hidesoftinput 以及当用户单击对话框外侧时软输入和对话框都关闭。

dialog = new Dialog(act) {
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        // Tap anywhere to close dialog.
        Rect dialogBounds = new Rect();
        getWindow().getDecorView().getHitRect(dialogBounds);
        if (!dialogBounds.contains((int) event.getX(),
                (int) event.getY())) {
            // You have clicked the grey area
            InputMethodManager inputMethodManager = (InputMethodManager) act
                    .getSystemService(act.INPUT_METHOD_SERVICE);
            inputMethodManager.hideSoftInputFromWindow(dialog
                    .getCurrentFocus().getWindowToken(), 0);
            dialog.dismiss();
            // stop activity closing
        } else {
            InputMethodManager inputMethodManager = (InputMethodManager) act
                    .getSystemService(act.INPUT_METHOD_SERVICE);
            inputMethodManager.hideSoftInputFromWindow(dialog
                    .getCurrentFocus().getWindowToken(), 0);
        }

        return true;
    }
};
于 2015-12-22T06:19:04.567 回答
3

另一种解决方案,此代码取自Window 您应该将这两个方法添加到您的对话框源代码的android源代码。

@Override
public boolean onTouchEvent(MotionEvent event) {        
    if (isShowing() && (event.getAction() == MotionEvent.ACTION_DOWN
            && isOutOfBounds(getContext(), event) && getWindow().peekDecorView() != null)) {
        hide();
    }
    return false;
}

private boolean isOutOfBounds(Context context, MotionEvent event) {
    final int x = (int) event.getX();
    final int y = (int) event.getY();
    final int slop = ViewConfiguration.get(context).getScaledWindowTouchSlop();
    final View decorView = getWindow().getDecorView();
    return (x < -slop) || (y < -slop)
            || (x > (decorView.getWidth()+slop))
            || (y > (decorView.getHeight()+slop));
}

这个解决方案没有这个问题:

这很好用,只是下面的活动也会对触摸事件做出反应。有什么方法可以防止这种情况吗?– 豪特尔

于 2014-07-07T13:54:57.727 回答
2

以下对我有用:

myDialog.setCanceledOnTouchOutside(true);
于 2019-12-05T08:56:59.187 回答
1

dialog.setCancelable(false);从您的活动/片段中调用。

于 2017-05-04T07:19:18.363 回答
0

您可以制作一个background占据所有屏幕大小transparent并收听onClick事件的dismiss内容。

于 2011-12-05T10:45:05.250 回答
0

我尝试了一些答案,但我仍然遇到了一个问题,比如当我在对话框之外按下时,对话框隐藏但显示了一个暗淡的视图,再次按下将转到父活动。但实际上我想在第一次点击后去父活动。所以我所做的是

dialog.setOnCancelListener(this);

并更改了我的活动以DialogInterface.OnCancelListener实施

@Override
public void onCancel(DialogInterface dialog) {
     finish();
}

繁荣,它奏效了。

于 2021-03-14T09:51:27.117 回答
-1

这是代码

    dialog.getWindow().getDecorView().setOnTouchListener(new View.OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent ev) {

            if(MotionEvent.ACTION_DOWN == ev.getAction())
            {
                Rect dialogBounds = new Rect();
                dialog. getWindow().getDecorView().getHitRect(dialogBounds);
                if (!dialogBounds.contains((int) ev.getX(), (int) ev.getY())) {
                    // You have clicked the grey area
                    UiUtils.hideKeyboard2(getActivity());
                    return false; // stop activity closing
                }
            }
            getActivity().dispatchTouchEvent(ev);
            return false;
        }
    });

试试这个。触摸外部时可以隐藏键盘

于 2020-03-13T11:59:52.003 回答