java - 如何在 JFileChooser 中的 "Enter"按钮上点击 "Cancel"后采取行动?

标签 java swing keyboard look-and-feel jfilechooser

我在 JFrame 中有一个 JFileChooser。我在 JFileChooser 中添加了一个 ActionListener,以便在单击时“取消”按钮起作用。我也可以切换到“取消”按钮,但是当我按下“Enter”键时,没有任何反应(即 ActionListener 没有使用事件命令 JFileChooser.CANCEL_SELECTION 调用)。我必须对 JFileChooser 做些什么才能使在“取消”按钮上按下“Enter”键等同于单击“取消”按钮?

这是我看到的(错误)行为的一个简单示例:

import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;

import javax.swing.JFileChooser;
import javax.swing.JFrame;

public final class TestApp {
    public static void main(final String[] args) {
        javax.swing.SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                try {
                    final JFileChooser chooser = new JFileChooser();
                    chooser.addActionListener(new ActionListener() {
                        public void actionPerformed(final ActionEvent e) {
                            System.exit(0);
                        }
                    });
                    final JFrame frame = new JFrame();
                    frame.add(chooser);
                    frame.pack();
                    frame.setVisible(true);
                }
                catch (final Throwable t) {
                    t.printStackTrace();
                }
            }
        });
    }
}

要查看(错误)行为,请执行程序,按 Tab 键选择“取消”,然后按“Enter”键。该程序不会在我的平台上终止——尽管当我单击“取消”按钮时它会终止。

扩展 JFileChooser 和覆盖 cancelSelection() 也不起作用(显然,在“取消”按钮)。

(错误)行为发生在我的 Fedora 10 x86_64 系统上,使用 Java 5、6 和 7。

附录:下面添加了一个 KeyEventPostProcessor 到当前的 KeyboardFocusManager 并且看起来像我想要的那样:

import java.awt.Component;
import java.awt.KeyEventPostProcessor;
import java.awt.KeyboardFocusManager;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyEvent;

import javax.swing.JButton;
import javax.swing.JFileChooser;
import javax.swing.JFrame;

public final class TestApp {
    public static void main(final String[] args) {
        javax.swing.SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                try {
                    final JFileChooser chooser = new JFileChooser();
                    chooser.addActionListener(new ActionListener() {
                        public void actionPerformed(final ActionEvent e) {
                            System.out.println(e.paramString());
                            System.exit(0);
                        }
                    });
                    final KeyboardFocusManager kfm = KeyboardFocusManager
                            .getCurrentKeyboardFocusManager();
                    kfm.addKeyEventPostProcessor(new KeyEventPostProcessor() {
                        @Override
                        public boolean postProcessKeyEvent(final KeyEvent e) {
                            if (e.getID() == KeyEvent.KEY_RELEASED
                                    && e.getKeyCode() == KeyEvent.VK_ENTER) {
                                final Component comp = e.getComponent();
                                if (chooser.isAncestorOf(comp)) {
                                    if (!(comp instanceof JButton)) {
                                        chooser.approveSelection();
                                    }
                                    else {
                                        final JButton button = (JButton) comp;
                                        if ("Cancel".equals(button.getText())) {
                                            chooser.cancelSelection();
                                        }
                                        else {
                                            chooser.approveSelection();
                                        }
                                    }
                                }
                            }
                            return false;
                        }
                    });
                    final JFrame frame = new JFrame();
                    frame.add(chooser);
                    frame.pack();
                    frame.setVisible(true);
                }
                catch (final Throwable t) {
                    t.printStackTrace();
                }
            }
        });
    }
}

然而,这似乎需要做很多工作,才能区分是在“取消”按钮上按下回车键还是在其他任何地方按下回车键。

你发现它有什么问题吗?

发现的解决方案:将 GUI 外观设置为我的系统 (Linux) 的 native 外观和感觉可以满足我的要求,无需任何其他操作。这就是我所不知道的,也是我一直在寻找的。解决方案是有以下内容

UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());

作为 main() 方法的第一个可执行语句。然后可以省去所有焦点监听器、关键事件处理器等。

我已将 100 分奖励给最有帮助的受访者。

最佳答案

The program doesn't terminate on my platform.

我看到 Mac OS X 10.5、Ubuntu 10 和 Windows 7 使用(各种)Java 5 和 6 正常运行。我用 println() 替换了你的 exit() > 查看事件:

System.out.println(rootDirChooser.getSelectedFile().getName() + e.paramString());

指定您的平台和版本可能会有所帮助;如果可能,还要验证安装是否正确。

我不确定我是否理解您的目标;但是,作为替代方案,请考虑覆盖 approveSelection():

private static class MyChooser extends JFileChooser {

    @Override
    public void approveSelection() {
        super.approveSelection();
        System.out.println(this.getSelectedFile().getName());
    }
}

附录:

The goal is to have the action of hitting the "Enter" key while on the "Cancel" button be identical to clicking on the "Cancel" button.

Key Bindings 中所述,您可以更改与 VK_ENTER 关联的操作。

KeyStroke enter = KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0);
InputMap map = chooser.getInputMap(JFileChooser.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT);
map.put(enter, "cancelSelection");

如果您希望更改仅在“取消”按钮具有焦点时发生,您需要在 Focus Listener 中执行此操作.

附录:

I found a solution that uses KeyboadFocusManager, instead. What do you think?

我可以看到每种方式的优缺点,所以我在下面概述了这两种方式。使用 KeyboadFocusManager 查找所有按钮,但不提供独立于语言环境的方法来区分它们; Focus Listener 方法只能看到批准按钮,并且它是特定于 UI 的。尽管如此,您还是可以结合使用这些方法以获得更好的结果。第二意见不会有问题。

附录:

我更新了下面的代码,不再需要知道“取消”按钮的本地化名称并使用键绑定(bind)。

import java.awt.EventQueue;
import java.awt.KeyboardFocusManager;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.awt.event.KeyEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.InputMap;
import javax.swing.JButton;
import javax.swing.JFileChooser;
import javax.swing.JFrame;
import javax.swing.KeyStroke;
import javax.swing.plaf.metal.MetalFileChooserUI;

public final class FileChooserKeys
    implements ActionListener, FocusListener, PropertyChangeListener {

    private final JFileChooser chooser = new JFileChooser();
    private final MyChooserUI myUI = new MyChooserUI(chooser);
    private final KeyStroke enterKey =
        KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0);

    private void create() {
        final JFrame frame = new JFrame();
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        chooser.addActionListener(this);
        myUI.installUI(chooser);
        myUI.getApproveButton(chooser).addFocusListener(this);
        KeyboardFocusManager focusManager =
            KeyboardFocusManager.getCurrentKeyboardFocusManager();
        focusManager.addPropertyChangeListener(this);

        frame.add(chooser);
        frame.pack();
        frame.setVisible(true);
    }

    @Override
    public void actionPerformed(ActionEvent e) {
        System.out.println(e.paramString());
    }

    @Override
    public void focusGained(FocusEvent e) {
        System.out.println("ApproveButton gained focus.");
    }

    @Override
    public void focusLost(FocusEvent e) {
        System.out.println("ApproveButton lost focus.");
    }

    @Override
    public void propertyChange(PropertyChangeEvent e) {
        Object o = e.getNewValue();
        InputMap map = chooser.getInputMap(
            JFileChooser.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT);
        if (o instanceof JButton) {
            if ("focusOwner".equals(e.getPropertyName())) {
                JButton b = (JButton) o;
                String s = b.getText();
                boolean inApproved = b == myUI.getApproveButton(chooser);
                if (!(s == null || "".equals(s) || inApproved)) {
                    map.put(enterKey, "cancelSelection");
                } else {
                    map.put(enterKey, "approveSelection");
                }
            }
        }
    }

    private static class MyChooserUI extends MetalFileChooserUI {

        public MyChooserUI(JFileChooser b) {
            super(b);
        }

        @Override
        protected JButton getApproveButton(JFileChooser fc) {
            return super.getApproveButton(fc);
        }
    }

    public static void main(final String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new FileChooserKeys().create();
            }
        });
    }
}

关于java - 如何在 JFileChooser 中的 "Enter"按钮上点击 "Cancel"后采取行动?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3378798/

相关文章:

java - 将应用程序移动到其他框架,然后返回到上一个

java - 防止 JWindow 出现在所有窗口的顶部

iOS:键盘框架更改通知错误?

java - CopyOnWriteArrayList 仅适用于迭代,不适用于随机访问读取

java - 使用 Eclipse IDE 在 Java 中设置类路径

java - GridBagLayout 似乎锚定在底部,并扩展标签

java - 更改背景颜色可编辑的 JComboBox

java - 显示时只有一小部分窗口可见。我如何展示整个事情?

swift - 如何在警报中设置键盘类型

keyboard - 检测VB6中存在的键盘