簡體   English   中英

Java KeyListener未檢測到鍵盤輸入

[英]Java KeyListener isn't detecting keyboard input

我正在嘗試修改此程序,以便繪制城堡的圖像,並且我可以使用向上和向下箭頭鍵縮放此圖像。 我無法讓keylistener工作,程序運行但它沒有響應按鍵。 任何幫助將不勝感激,謝謝。

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--;
        }
    }
}

我無法讓keylistener工作,程序運行但它沒有響應按鍵。

你的問題是經常被問到的一個非常常見的問題,幾乎總是問題是由於缺乏關注 - 被收聽的組件沒有當前的焦點,並且焦點對於KeyListener的工作至關重要。

簡短的回答是給予正在聽取焦點的組件。

更好的答案是不使用KeyListeners,而是使用Key Bindings來實現這樣的項目。


編輯
其他問題:

  • 在上面的代碼中,您將KeyListener添加到JFrame,即使您有充分的理由使用KeyListeners,也不應該這樣做。
  • 你也直接在JFrame中繪圖,在我看來這是一個更大的問題,因為你冒着不必要的副作用。
  • 你在paint方法中添加了一個監聽器,這是一個更大的問題。 不應該覆蓋paint方法(通常),如果你必須覆蓋它,則永遠不應該用於程序邏輯,添加或刪除偵聽器,或者進行任何非繪制活動。 它應該只用於繪畫和繪畫。
  • 相反,您應該直接在JPanel或JComponent中繪制。
  • 相反,您應該在繪畫JPanel或JComponent中的paintComponent(Graphics g)覆蓋中進行繪制。
  • 如上所述,此方法應僅用於繪畫和繪畫,並且應盡可能快。
  • 您應該在paintComponent(Graphics g)覆蓋內部調用super.paintComponent(g)
  • 繪圖組件應該再次使用Key Bindings來監聽擊鍵( 這里的教程)。 本教程將解釋為什么這種區別(KeyListener與Key Bindings)很重要。

編輯2
你永遠不想忽略異常,因為你的代碼在這里顯示,因為這是驅動盲人的編程等價物:

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

希望這不是你的生產代碼的外觀,你只是為了簡潔而忽略了你發布的代碼中的異常,但是我們很多人都明白這就像在粉筆板上聽到釘子一樣。


編輯3
有關KeyListeners與Key Bindings的更多信息:假設您的代碼與KeyListener一起使用,那么假設您向GUI添加了任何其他可聚焦組件,並且它們通過用戶交互以某種方式獲得了焦點,那么您的KeyBindings將不再起作用。 如果您使用Key Bindings正確完成了這一操作,那么這不是問題。


編輯4
你真的希望你的scale字段是double,而不是int。 並且你真的不想增加和減少它,而是你想要乘以它並將它除以一些乘數常數,比如1.2作為例子。 每當您更改比例時,您還需要調用repaint()


編輯5
請查看兩個示例程序,第一個名為DrawImagePanelKeyListener.java,使用KeyListener,而第二個名為DrawImagePanelKeyBindings,使用Key Bindings。 它們都應按預期編譯,運行和運行:當您按向上或向下箭頭鍵時,顯示的圖像會縮小並增大。 當按下JButton時,可以看到它們的行為差異。 按下按鈕,查看您的密鑰響應會發生什么。 當具有KeyListener的組件失去焦點時,其KeyListener將停止工作,但對於使用鍵綁定的組件則不同。

可以解決這個問題的一個問題可能是阻止所有其他組件獲得焦點,但對於大多數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();
   }
}

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM