如何在用户点击JDialog之外时关闭模态JDialog?

时间:2009-11-04 19:02:21

标签: java swing modal-dialog jdialog

我有一个Undecorated Modal JDialog,我想在用户点击模态对话框之外时设置为可见(false)。

这可以在Swing中使用吗?

我正在做的是为日期选择器等文本字段弹出自定义编辑器。有没有更简单的方法来做我想要的事情?

修改

请记住调用setVisible(true)时的模态块,所以你不能只说“不要使用模态对话框”

我已经尝试在对话框上集中监听器,它们在模态时不会触发。

7 个答案:

答案 0 :(得分:11)

编辑: 更改为使用WindowFocusListener而不是FocusListener,并检查丢失焦点的降序组件,以便在子组件获得焦点时不隐藏。

一种简单的方法是在对话框上添加一个窗口焦点侦听器,在焦点丢失时隐藏它。在这种情况下,我认为不需要模态。例如:

import javax.swing.*;
import java.awt.*;
import java.awt.event.WindowEvent;
import java.awt.event.WindowFocusListener;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;

public class ClickAwayDialog extends JDialog {

    public ClickAwayDialog(final Frame owner) {
        super(owner);
        JPanel pnl = new JPanel(new BorderLayout());
        pnl.add(new JLabel("Click outside this dialog in the parent frame to close it"), BorderLayout.NORTH);
        JButton btn = new JButton("Click Me");
        btn.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent e) {
                JOptionPane.showMessageDialog(ClickAwayDialog.this, "New Child Window");
            }
        });
        pnl.add(btn, BorderLayout.CENTER);
        this.setContentPane(pnl);
        this.pack();
        this.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
        this.setLocationRelativeTo(owner);
        this.setAlwaysOnTop(true);
        this.addWindowFocusListener(new WindowFocusListener() {

            public void windowGainedFocus(WindowEvent e) {
                //do nothing
            }

            public void windowLostFocus(WindowEvent e) {
                if (SwingUtilities.isDescendingFrom(e.getOppositeWindow(), ClickAwayDialog.this)) {
                    return;
                }
                ClickAwayDialog.this.setVisible(false);
            }

        });
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                JFrame parent = new JFrame();
                parent.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
                parent.setSize(300, 300);
                parent.setLocationByPlatform(true);
                parent.setVisible(true);
                ClickAwayDialog dlg = new ClickAwayDialog(parent);
                dlg.setVisible(true);                
            }
        });
    }
}

答案 1 :(得分:4)

如果您可以在其外部点击并发生“某些事情”,那么这不是模态对话框。所有答案都是正确的,您应该创建一个非模态对话框,然后通过FocusListener处理您的用例。

答案 2 :(得分:2)

没有必要成为模态对话框(模态意味着它会阻止您在隐藏对话框之前使用所有者窗口)。最好试试这个:

final JDialog dlg ...
dlg.setModal(false);

dlg.addWindowFocusListener(new WindowFocusListener() {            
    public void windowLostFocus(WindowEvent e) {
        dlg.setVisible(false);
    }            
    public void windowGainedFocus(WindowEvent e) {
    }
});

答案 3 :(得分:1)

尝试将模态设置为false,然后使用windowsDeactivated()进行关闭对话框(dialog.dispose()),对我有用。

答案 4 :(得分:0)

不是真正的模态对话框,如果点击其他地方关闭它,也许你想要setAlwaysOnTop

然而,类似下面的内容应该可以做到(未经测试)。请注意,我建议将代码移动到比设备使用更好的设计。

static JDialog dialog = ...

Toolkit.getDefaultToolkit().addAWTEventListener(new AWTEventListener() {
    public void eventDispatched(AWTEvent e) {
        dialog.setVisible(false);

        SwingUtils.invokeLater(new Runnable(){
            public void run(){
                Toolkit.getDefaultToolkit().removeAWTEventListener(this);
            }
        });        
    }
}, AWTEvent.MOUSE_EVENT_MASK);

dialog.setVisible(true);

答案 5 :(得分:0)

可能会添加FocusListener并在失去焦点时隐藏对话框。如果对话框中的某些元素可以具有焦点,则可能会很棘手。无论如何,试验它。

答案 6 :(得分:0)

使用WindowListener并处理windowDeactivated()事件。