我无法让 keylistener 工作,程序运行但它没有响应按键。
你的问题是一个非常常见的问题,在这里经常被问到,而且几乎总是由于缺乏焦点——被监听的组件没有当前焦点,而焦点对于 KeyListener 的工作至关重要。
简短的回答是让正在收听的组件成为焦点。
更长更好的答案是不使用 KeyListeners 而是使用 Key Bindings 来进行这样的项目。
编辑
其他问题:
- 在上面的代码中,您将 KeyListener 添加到 JFrame,即使您有充分的理由使用 KeyListener,也不应该这样做。
- 您还直接在 JFrame 中绘图,在我看来这是一个更大的问题,因为您冒着不必要的副作用的风险。
- 您在paint 方法中添加了一个侦听器,这是一个更大的问题。paint 方法不应该被覆盖(一般来说),如果你必须覆盖它,则永远不应该用于程序逻辑、添加或删除侦听器或执行任何非绘画活动。它应该只用于绘画和绘画。
- 相反,您应该直接在 JPanel 或 JComponent 中绘制。
- 相反,您应该
paintComponent(Graphics g)
在您的绘画 JPanel 或 JComponent 中进行覆盖。
- 如上所述,这种方法应该只用于绘画和绘画,并且应该尽可能快。
- 您应该调用覆盖的
super.paintComponent(g)
内部。paintComponent(Graphics g)
- 绘图组件应再次使用 Key Bindings 监听击键(此处为教程)。本教程将解释为什么这种区别(KeyListener 与 Key Bindings)很重要。
编辑 2
您永远不想忽略异常,因为您的代码在此处显示,因为这相当于盲目驾驶的编程:
try {
url = this.getClass().getResource(filename);
} catch (Exception e) {
}
希望这不是您的生产代码的外观,为了简洁起见,您只是忽略了发布代码中的异常,但对于我们许多人来说,理解这就像在黑板上听到钉子。
编辑 3
关于 KeyListener 与 Key Bindings 的更多信息:假设您的代码可以使用 KeyListener,然后假设您将任何其他可聚焦组件添加到 GUI 并且它们以某种方式通过用户交互获得了焦点,那么您的 KeyBindings 将不再工作。如果您使用键绑定正确地做到了这一点,这将不是问题。
编辑 4
您真的希望您的比例字段是双精度,而不是整数。而且您真的不想增加和减少它,而是想将它乘以并除以某个乘数常数,例如 1.2。repaint()
每当你改变你的规模时,你也会想打电话。
编辑 5
请查看两个示例程序,第一个名为 DrawImagePanelKeyListener.java,使用 KeyListener,而第二个名为 DrawImagePanelKeyBindings,使用 Key Bindings。它们都应该按预期编译、运行和运行:当您按向上或向下箭头键时,显示的图像会缩小和放大。但是,当 JButton 都被按下时,可以看到它们的行为差异。按下按钮,看看您的按键响应会发生什么。当具有 KeyListener 的组件失去焦点时,其 KeyListener 将停止工作,但对于使用 Key Bindings 的组件则不然。
可以解决这个问题的一个办法可能是阻止所有其他组件获得焦点,但这对于大多数 GUI 来说是不切实际或不希望的。
DrawImagePanelKeyListener.java
import java.awt.*;
import javax.imageio.ImageIO;
import javax.swing.*;
import java.io.IOException;
import java.net.*;
import java.awt.event.*;
@SuppressWarnings("serial")
public class DrawImagePanelKeyListener extends JPanel implements KeyListener {
public static final String IMAGE_PATH = "https://duke.kenai.com/"
+ "nyanya/.Midsize/NyaNya.jpg.png";
private static final double MULTIPLIER = 1.2;
double scale = 1.0;
private Image image;
private Dimension initialSize = new Dimension(0, 0);
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
DrawImagePanelKeyListener drawImage = new DrawImagePanelKeyListener();
drawImage.add(new JButton("Foo"));
JFrame frame = new JFrame("Draw Image");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(drawImage);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public DrawImagePanelKeyListener() {
setFocusable(true);
requestFocusInWindow();
URL imgURL;
try {
imgURL = new URL(IMAGE_PATH);
image = ImageIO.read(imgURL);
initialSize = new Dimension(image.getWidth(this),
image.getHeight(this));
addKeyListener(this);
setVisible(true);
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
@Override
public Dimension getPreferredSize() {
return initialSize;
}
@Override
protected void paintComponent(Graphics g) {
super.paintComponent(g);
g.setColor(Color.BLACK);
g.fillRect(0, 0, getSize().width, getSize().height);
Graphics2D g2d = (Graphics2D) g.create();
g2d.scale(scale, scale);
g2d.drawImage(image, 0, 0, this);
}
public void keyReleased(KeyEvent e) {
}
public void keyTyped(KeyEvent e) {
}
public void keyPressed(KeyEvent e) {
int key = e.getKeyCode();
if (key == KeyEvent.VK_UP) {
scale *= MULTIPLIER;
}
if (key == KeyEvent.VK_DOWN) {
scale /= MULTIPLIER;
}
repaint();
}
}
DrawImagePanelKeyBindings.java
import java.awt.*;
import javax.imageio.ImageIO;
import javax.swing.*;
import java.io.IOException;
import java.net.*;
import java.awt.event.*;
@SuppressWarnings("serial")
public class DrawImagePanelKeyBindings extends JPanel {
public static final String IMAGE_PATH = "https://duke.kenai.com/"
+ "nyanya/.Midsize/NyaNya.jpg.png";
private static final double MULTIPLIER = 1.2;
double scale = 1.0;
private Image image;
private Dimension initialSize = new Dimension(0, 0);
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
DrawImagePanelKeyBindings drawImage = new DrawImagePanelKeyBindings();
drawImage.add(new JButton("Foo"));
JFrame frame = new JFrame("Draw Image");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.add(drawImage);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public DrawImagePanelKeyBindings() {
setBindings();
URL imgURL;
try {
imgURL = new URL(IMAGE_PATH);
image = ImageIO.read(imgURL);
initialSize = new Dimension(image.getWidth(this),
image.getHeight(this));
setVisible(true);
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
private void setBindings() {
int condition = WHEN_IN_FOCUSED_WINDOW;
InputMap inputMap = getInputMap(condition);
ActionMap actionMap = getActionMap();
final KeyStroke[] keyStrokes = {
KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0),
KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0)
};
for (final KeyStroke keyStroke : keyStrokes) {
inputMap.put(keyStroke, keyStroke.toString());
actionMap.put(keyStroke.toString(), new AbstractAction() {
@Override
public void actionPerformed(ActionEvent evt) {
myKeyPressed(keyStroke.getKeyCode());
}
});
}
}
@Override
public Dimension getPreferredSize() {
return initialSize;
}
@Override
protected void paintComponent(Graphics g) {
super.paintComponent(g);
g.setColor(Color.BLACK);
g.fillRect(0, 0, getSize().width, getSize().height);
Graphics2D g2d = (Graphics2D) g.create();
g2d.scale(scale, scale);
g2d.drawImage(image, 0, 0, this);
}
public void myKeyPressed(int key) {
if (key == KeyEvent.VK_UP) {
scale *= MULTIPLIER;
}
if (key == KeyEvent.VK_DOWN) {
scale /= MULTIPLIER;
}
repaint();
}
}