当前位置: 首页 > 面试题库 >

使用KeyAdapter的简单程序无法在按下多个按键后注册单个按键。代码解决方案?

谯振国
2023-03-14
问题内容

我只是在尝试简单绘图程序的基础知识。我有一个程序可以绘制一个矩形,然后让我向任意方向移动它。我使用线程进行平滑操作。但是,令我困扰的是,如果我按下三个键并释放其中两个,程序将不会检测到第三个,从而使矩形保持静止。

我在这里要求太多,还是与硬件有关?我有一个很便宜的键盘。

如果您想兜风,这里是完整的程序。

import java.awt.Color;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;

import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;

public class Boxxy extends JFrame implements Runnable{

    int xPos, yPos, xDir, yDir;
    MyPanel panel;
    ImageIcon ii;
    Image i;


    public Boxxy() {
        setSize(400,400);
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setLocationRelativeTo(null);

        panel = new MyPanel();
        add(panel);

        setVisible(true);
    }

    class MyPanel extends JPanel {
        private static final long serialVersionUID = 1L;

        private int squareX = 0;
        private int squareY = 0;
        private int squareW = 20;
        private int squareH = 20;

        private MyPanel() {
            setFocusable(true);
            addKeyListener(new KeyAdapter() {
                public void keyPressed(KeyEvent e) {

                    switch (e.getKeyCode()) {
                    case (KeyEvent.VK_UP): setyDir(-1);
                    break;
                    case (KeyEvent.VK_DOWN): setyDir(1);
                    break;
                    case (KeyEvent.VK_LEFT): setxDir(-1);
                    break;
                    case (KeyEvent.VK_RIGHT): setxDir(1);
                    break;
                    }
                }

                public void keyReleased(KeyEvent e) {
                    switch (e.getKeyCode()) {
                    case (KeyEvent.VK_UP): setyDir(0);
                    break;
                    case (KeyEvent.VK_DOWN): setyDir(0);
                    break;
                    case (KeyEvent.VK_LEFT): setxDir(0);
                    break;
                    case (KeyEvent.VK_RIGHT): setxDir(0);
                    break;
                    }
                }
            });
        }

        private void repaintSquare(int x, int y) {
            int OFFSET = 1;
            if ((squareX!=x) || (squareY!=y)) {
                repaint(squareX,squareY,squareW+OFFSET,squareH+OFFSET);
                squareX=x;
                squareY=y;
                repaint(squareX,squareY,squareW+OFFSET,squareH+OFFSET);
            }
        }

        protected void paintComponent(Graphics g) {
            super.paintComponent(g);  
            g.setColor(Color.RED);
            g.fillRect(squareX,squareY,squareW,squareH);
        }
    }

    private void move() {
        panel.repaintSquare(xPos, yPos);
        xPos += xDir;
        yPos += yDir;
    }

    private void setxDir(int x) {
        this.xDir = x;
    }

    private void setyDir(int y) {
        this.yDir = y;
    }

    public static void main(String[] args) {
        Thread t = new Thread(new Boxxy());
        t.start();
    }

    @Override
    public void run() {
        try {
            while(true) {
                move();
                Thread.sleep(5);
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

问题答案:

正如在类似问题中所讨论的,不要使用KeyListener,而要使用键绑定。摆动计时器可用于重复移动精灵。

编辑
例如:

import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.event.*;
import java.util.EnumMap;

import javax.swing.*;

@SuppressWarnings("serial")
public class Boxxy2 extends JPanel {
   private static final int PREF_W = 400;
   private static final int PREF_H = PREF_W;
   private static final int SQUARE_W = 20;
   private static final String PRESSED = "pressed";
   private static final String RELEASED = "released";
   public static final int MOVE_SCALE = 1;
   private static final int TIMER_DELAY = 5;
   private int squareX = 0;
   private int squareY = 0;
   private int squareW = SQUARE_W;
   private int squareH = SQUARE_W;
   private EnumMap<Direction, Boolean> dirMap = new EnumMap<>(Direction.class);

   public Boxxy2() {
      setupKeyBinding();
      new Timer(TIMER_DELAY, new TimerListener()).start();
   }

   private void setupKeyBinding() {
      int condition = JComponent.WHEN_IN_FOCUSED_WINDOW;
      InputMap inMap = getInputMap(condition);
      ActionMap actMap = getActionMap();

      // this uses an enum of Direction that holds ints for the arrow keys
      for (Direction direction : Direction.values()) {
         dirMap.put(direction, Boolean.FALSE);
         int key = direction.getKey();
         String name = direction.name();

         KeyStroke keyPressed = KeyStroke.getKeyStroke(key, 0, false);
         KeyStroke keyReleased = KeyStroke.getKeyStroke(key, 0, true);

         // add the key bindings for arrow key and shift-arrow key
         inMap.put(keyPressed, name + PRESSED);
         inMap.put(keyReleased, name + RELEASED);
         actMap.put(name + PRESSED, new MyKeyAction(this, direction, true));
         actMap.put(name + RELEASED, new MyKeyAction(this, direction, false));
      }
   }

   @Override
   public Dimension getPreferredSize() {
      return new Dimension(PREF_W, PREF_H);
   }

   @Override
   protected void paintComponent(Graphics g) {
      super.paintComponent(g);
      g.setColor(Color.RED);
      g.fillRect(squareX, squareY, squareW, squareH);
   }

   private static void createAndShowGui() {
      Boxxy2 mainPanel = new Boxxy2();

      JFrame frame = new JFrame("Boxxy2");
      frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
      frame.getContentPane().add(mainPanel);
      frame.pack();
      frame.setLocationByPlatform(true);
      frame.setVisible(true);
   }

   public static void main(String[] args) {
      SwingUtilities.invokeLater(new Runnable() {
         public void run() {
            createAndShowGui();
         }
      });
   }

   private class TimerListener implements ActionListener {
      @Override
      public void actionPerformed(ActionEvent e) {
         for (Direction dir : dirMap.keySet()) {
            if (dirMap.get(dir)) {
               squareX += MOVE_SCALE * dir.getRight();
               squareY += MOVE_SCALE * dir.getDown();
            }
         }
         repaint();
      }
   }

   public void setMovement(Direction direction, boolean keyPressed) {
      dirMap.put(direction, keyPressed);
   }
}

enum Direction {
   UP(KeyEvent.VK_UP, 0, -1), DOWN(KeyEvent.VK_DOWN, 0, 1), LEFT(KeyEvent.VK_LEFT, -1, 0), RIGHT(KeyEvent.VK_RIGHT, 1, 0);

   private int key;
   private int right;
   private int down;

   private Direction(int key, int right, int down) {
      this.key = key;
      this.right = right;
      this.down = down;
   }

   public int getKey() {
      return key;
   }

   public int getRight() {
      return right;
   }

   public int getDown() {
      return down;
   }

}

// Actions for the key binding
@SuppressWarnings("serial")
class MyKeyAction extends AbstractAction {
   private Boxxy2 boxxy2;
   private Direction direction;
   private boolean keyPressed;

   public MyKeyAction(Boxxy2 boxxy2, Direction direction, boolean keyPressed) {
      this.boxxy2 = boxxy2;
      this.direction = direction;
      this.keyPressed = keyPressed;
   }

   @Override
   public void actionPerformed(ActionEvent e) {
      boxxy2.setMovement(direction, keyPressed);
   }
}


 类似资料:
  • 问题内容: 是否有可能结合按键的组合来触发单个事件? 我已经在Chrome浏览器中尝试过,但该事件不会触发。 称我为疯狂,但我正在编写Chrome扩展程序,并希望同时按下+ + 键将其强制为隐藏的开发人员模式。 问题答案: 如果要检测的,和钥匙都在同一时间全部下来,你得看都和并保持地图为down的那些的。当他们全都崩溃时,请触发您的事件。 我假设您不关心它们按什么顺序按下(因为要可靠地按一下会很痛

  • 问题内容: 我有以下方法可以在Angular中捕获按键: 我听着 但是,我想检测何时同时按下两个键,例如和同时按下(不是一个一个又一个又一个的组合)。 最好的方法是什么? 编辑 我当时想的是: 因此,如果我有多个按键,那么我将创建正确的。但是,问题在于,现在顺序很重要(即,如果我按then ,则按is ;如果我按另一种方式按,则得到) 问题答案: 我认为广播使用过多。而是使用自定义指令?这是一个用

  • 问题内容: 我有一个带有数字值的字符串键数组,可用于具有每个标签出现次数的标签列表中,因此: 这是为了我可以按降序显示标签列表,因此: 我可以使用 arsort 通过出色的值进行反向排序,但是我还希望具有相同数字值的所有标签都按字母顺序排序,因此最终结果可以是: 有办法可以做到吗?我猜想 usort 可能是要走的路,但是我看了php.net上的示例,我的眼睛呆呆了!非常感谢!!! 问题答案: 看一

  • Java中的公共类 我希望在键盘上按Q会使控制台打印,但它只是垃圾邮件和

  • 问题内容: 我想按一下软键盘上的任何键。我不想在Activity中使用EditView或TextView,必须从Activity内部的扩展View处理该事件。 我只是试过这个: 1)覆盖活动方法。该功能不适用于软键盘,只能捕获很少的硬键盘。 2)创建我的并在我的视图中注册该视图,其中包含已注册并正在运行的。这对于软键盘根本不起作用。 3)覆盖View方法。如果我设置我的OnKeyListener或

  • 问题内容: 我有一个使用javascript进行游戏的想法(我将使用EaselJS进行开发),我将不得不检测按键。在互联网上四处浏览后,我看到了很多建议(使用window.onkeypress,使用jQuery等),但是几乎每个选项都有一个反对意见。你们有什么建议?使用jQuery听起来很容易,但实际上我没有使用该库的经验(而且我也不是JavaScript的资深人士),所以我宁愿避免使用它。如果j