java - 为什么一个 Swing 击键有效,而另一个无效?
问题描述
我正在编写一个具有 JTable 的 Swing 程序;我想使用 control-V 粘贴到 JTable 中,并使用 control-S 保存 JTable 上的信息。
我第一次使用JTable.registerKeyboardAction()
control-V 键注册动作,这很有效,但我注意到它的 javadoc 说它已经过时,新代码应该为此目的使用输入映射和动作映射。
我已经将这些用于 control-S 键,我也将其映射到了 JButton,所以我认为它很容易复制。这是我现在用于创建 JFrame 的代码片段:
private JFrame createMainframe()
{
JFrame frame = new JFrame("VisaExtraction");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
String saveFileActionName = "Save";
String pasteActionName = "Paste";
Action saveFileAction = new SaveFileAction(saveFileActionName, frame, tableModel);
Action pasteAction = new PasteAction(pasteActionName, frame, tableModel);
// JButton saveButton = new JButton(saveFileAction);
// saveButton.setMnemonic(KeyEvent.VK_S);
JPanel topPanel = new JPanel();
// topPanel.add(saveButton);
mainTable = new LastColumnChangesWidthJTable(tableModel);
JScrollPane scrollPane = new JScrollPane(mainTable);
// set ctrl-s to the 'saveFile' action
// and ctrl-v to the 'paste' action
InputMap tableInputMap = mainTable.getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW);
KeyStroke saveKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_S, InputEvent.CTRL_DOWN_MASK);
tableInputMap.put(saveKeystroke, saveFileActionName);
KeyStroke pasteKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_V, InputEvent.CTRL_DOWN_MASK);
tableInputMap.put(pasteKeystroke, pasteActionName);
// set the saveFile and paste actions to be executed when invoked.
ActionMap tableActionMap = mainTable.getActionMap();
tableActionMap.put(saveFileActionName, saveFileAction);
tableActionMap.put(pasteActionName, pasteAction);
frame.add(topPanel, BorderLayout.NORTH);
frame.add(scrollPane, BorderLayout.CENTER);
frame.pack();
// register ctrl-v to paste into the JTable
// mainTable.registerKeyboardAction
// ( actionListener -> handlePaste(tableModel, new VisaExtractionListener(tableModel)),
// KeyStroke.getKeyStroke(KeyEvent.VK_V, InputEvent.CTRL_DOWN_MASK),
// JComponent.WHEN_IN_FOCUSED_WINDOW
// );
return frame;
}
正如你所看到的,我用“S”和“save”做同样的事情,我用“V”和“paste”做,但是在我启动程序后,“control-S”工作(到告诉我没有什么可以保存的程度)并且“control-V”没有(actionPerformed()
方法中的断点没有被命中)。
什么可能导致这两者之间的差异?
解决方案
您的问题可能是由于 JTable 已经在另一个输入映射中使用了 ctrl-V 操作。了解组件有 3 个输入映射,我相信一个 forJComponent.WHEN_FOCUSED
优先于WHEN_IN_FOCUSED_WINDOW
.
我的 MCVE 用于概念验证。更改以下代码中的注释字段以查看哪个有效:
import java.awt.event.ActionEvent;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import javax.swing.*;
public class Foo {
public static void main(String[] args) {
Integer[][] rowData = {{1, 2}, {3, 4}};
String[] columnNames = {"A", "B"};
JTable table = new JTable(rowData, columnNames);
JScrollPane scrollPane = new JScrollPane(table);
// int condition = JComponent.WHEN_IN_FOCUSED_WINDOW;
int condition = JComponent.WHEN_FOCUSED;
InputMap inputMap = table.getInputMap(condition);
ActionMap actionMap = table.getActionMap();
KeyStroke saveKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_S, InputEvent.CTRL_DOWN_MASK);
KeyStroke pasteKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_V, InputEvent.CTRL_DOWN_MASK);
inputMap.put(saveKeystroke, saveKeystroke.toString());
inputMap.put(pasteKeystroke, pasteKeystroke.toString());
actionMap.put(saveKeystroke.toString(), new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
System.out.println("save action");
}
});
actionMap.put(pasteKeystroke.toString(), new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
System.out.println("paste action");
}
});
JFrame frame = new JFrame("Foo");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(scrollPane);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
}
同样在未来,创建 MCVE 的工作应该是你的,因为你是寻求帮助的人。
编辑:我错了。已经在使用并让您感到困惑的输入地图是JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT
地图。这在我的 MCVE 更新中为 control-V 键提供了一个非空键字符串:
import java.awt.event.ActionEvent;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import javax.swing.*;
public class Foo {
public static void main(String[] args) {
SwingUtilities.invokeLater(() -> createGui());
}
public static void createGui() {
Integer[][] rowData = { { 1, 2 }, { 3, 4 } };
String[] columnNames = { "A", "B" };
JTable table = new JTable(rowData, columnNames);
JScrollPane scrollPane = new JScrollPane(table);
int condition = JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT;
// int condition = JComponent.WHEN_IN_FOCUSED_WINDOW;
// int condition = JComponent.WHEN_FOCUSED;
InputMap inputMap = table.getInputMap(condition);
ActionMap actionMap = table.getActionMap();
KeyStroke saveKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_S, InputEvent.CTRL_DOWN_MASK);
KeyStroke pasteKeystroke = KeyStroke.getKeyStroke(KeyEvent.VK_V, InputEvent.CTRL_DOWN_MASK);
String pasteKey = (String) inputMap.get(pasteKeystroke);
System.out.println(pasteKey);
inputMap.put(saveKeystroke, saveKeystroke.toString());
inputMap.put(pasteKeystroke, pasteKeystroke.toString());
actionMap.put(saveKeystroke.toString(), new MyAction("Save Action"));
actionMap.put(pasteKeystroke.toString(), new MyAction("Paste Action"));
JFrame frame = new JFrame("Foo");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(scrollPane);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
}
@SuppressWarnings("serial")
class MyAction extends AbstractAction {
private String text;
public MyAction(String text) {
this.text = text;
}
@Override
public void actionPerformed(ActionEvent e) {
System.out.println(text);
}
}
推荐阅读
- c - 使用 Frama-C 检查 C 代码是否存在无效的内存访问
- flutter - 增加小部件的点击检测区域?
- python - 更改从 *.csv 文件读取的数据
- sass - 是否可以将 Scss 转换为 Material-UI 样式格式?
- python - 使用 asyncssh 在远程服务器上运行命令
- sql - 一行中每个人的最后 3 个就业信息
- javascript - 表单提交不适用于validationSchema
- javascript - Angular 2 - 在最近的元素上设置 ToggleClass
- php - require() 不工作 - NGINX PHP 配置问题
- linux - 如何在 CentOS 上刷新会话而不注销/登录?