Java KeyListener无法检测到键盘输入。

5

我正在尝试修改这个程序,使它绘制一幅城堡图片,我可以使用上下箭头键来缩放这个图像。然而,我无法让键盘监听器正常工作,程序运行时并未响应按键操作。如果有帮助的话,将不胜感激。谢谢。

import java.awt.*;
import java.awt.geom.*;
import java.util.*;
import javax.swing.*;
import java.net.*;
import java.awt.event.*;

public class DrawImage extends JFrame implements KeyListener {
int scale = 1;
    private Image image;
    enter code here
    public static void main(String[] args) {
        new DrawImage();
    }

    public DrawImage() {
        super("DrawImage");
        setSize(600,600);
        setVisible(true);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        Toolkit tk = Toolkit.getDefaultToolkit();
        image = tk.getImage(getURL("castle.png"));
        addKeyListener(this);
    }

    private URL getURL(String filename) {
        URL url = null;
        try {
            url = this.getClass().getResource(filename);
        }
        catch (Exception e) { }
        return url;
    }

    public void paint(Graphics g) {
        Graphics2D g2d = (Graphics2D) g;
        AffineTransform trans = new AffineTransform();
        trans.scale(scale, scale);
        System.out.println("scale: " + scale);
        g2d.setColor(Color.BLACK);
        g2d.fillRect(0, 0, getSize().width, getSize().height);
        g2d.setTransform(trans);
        g2d.drawImage(image, 0, 40, this);
        addKeyListener(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++;
        }
        if (key == KeyEvent.VK_DOWN) {
            scale--;
        }
    }
}

请查看编辑1、2、3和4以回答问题。如果有任何不清楚的地方,请留言说明。 - Hovercraft Full Of Eels
1个回答

8
我无法使键盘监听器起作用,程序可以运行但无法响应按键。这是一个常见问题,经常在这里被问到,几乎总是由于缺乏焦点导致的--即被监听的组件没有当前焦点,而焦点对于KeyListener的工作是必要的。简短的答案是给被监听的组件焦点。更好的长答案是不要使用KeyListener,而是对于这样的项目使用Key Bindings。

编辑
其他问题:

  • 在你上面的代码中,你将KeyListener添加到了JFrame上,这样做是不应该的,即使你有充分的理由使用KeyListener。
  • 你还直接在JFrame中进行绘图,我认为这是一个更大的问题,因为你可能会遇到意想不到的副作用。
  • 你在paint方法中添加了一个监听器,这是一个更大的问题。通常情况下,不应该覆盖paint方法,如果你确实需要覆盖它,则不应该用于程序逻辑、添加或删除监听器或执行任何非绘画活动。它应该只用于绘画。
  • 相反,你应该直接在JPanel或JComponent中绘制。
  • 相反,你应该在绘画JPanel或JComponent中重写paintComponent(Graphics g)方法进行绘制。
  • 就像上面所描述的那样,这个方法应该只用于绘画,并且应该尽可能快。
  • 你应该在你的paintComponent(Graphics g)重写中调用super.paintComponent(g)
  • 同样,绘画组件应该使用键绑定(Key Bindings)监听按键事件(教程here)。本教程将解释为什么这种区别(KeyListener vs. Key Bindings)很重要。

编辑2
您永远不希望忽略异常,因为这就像盲目驾驶一样:

  try {
     url = this.getClass().getResource(filename);
  } catch (Exception e) {
  }

希望这不是您的生产代码的样子,您只是为了简洁而忽略了发布的代码中的异常,但对于我们中的许多人来说,看到这就像听到划着指甲在黑板上发出的声音。


编辑3
关于KeyListeners和Key Bindings的更多信息:假设您已经使用KeyListeners使您的代码工作,然后假设您添加了任何其他可聚焦组件到您的GUI,并且它们通过用户交互以某种方式获得了焦点,则您的KeyBindings将不再起作用。如果您正确使用了Key Bindings,则这将不是一个问题。


编辑4
您真正希望您的比例字段是double而不是int。而且您真的不想增加和减少它,而是想乘以某个乘数常量,比如1.2作为一个例子。每当您更改比例时,您还需要调用repaint()


请查看两个示例程序,第一个名为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();
   }
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接