KeyListener в JPanel случайно не отвечает

#java #swing #jpanel #keylistener

#java #swing #jpanel #keylistener

Вопрос:

У меня возникли проблемы с Java KeyListener по умолчанию в моем проекте. Я заметил, что KeyListener, похоже, иногда не перенаправляет ключевые события при запуске.

Симптомы проблемы: При запуске приложения ввод ключа не обрабатывается. Это случается только иногда. Иногда мне приходится закрывать и запускать приложение 7-8 раз, пока это не появится. Иногда это первая попытка. Когда это произойдет, это не будет работать, пока я снова не перезапущу приложение.

Что я использую: Window 7 x64 и новейшие версии Eclipse и JDK.

Что я уже выяснил: я перевел точку останова в режим отладки и проверил экземпляр JPanel. Похоже, что KeyListener всегда успешно добавляется к нему. Кроме того, MouseListener и MouseMotionListener работают просто отлично, постоянно.

Минимальный код:

 public class Player implements KeyListener
{
    public void keyTyped(KeyEvent e){}
    public void keyReleased(KeyEvent e){ }

    public void keyPressed(KeyEvent e){
        System.out.println("Key Pressed!");
    }

}

public class Game {

    public static void main(String[] args) {
        new Game();
    } 

    public Game(){
        JFrame window = new JFrame();
        window.setVisible(true);

        //Now set the window size correctly
        window.setSize(800, 600);  
        //Set-up the rest of the window
        window.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
        window.setResizable(true);


        //Create our panel
        JPanel canvas = new JPanel();
        canvas.setFocusable(true);
        window.add( canvas ); //Add it to our window

        Player k = new Player();
        canvas.addKeyListener(k);
    }
}
  

Спасибо, что уделили мне время!

PS: Хорошо, ответьте на мой собственный вопрос:

Кажется, что я должен вызвать setVisible (true) после установки размера окна:

     JFrame window = new JFrame();


    Now set the window size correctly
    window.setSize(800, 600);  
    window.setVisible(true);
  

Похоже, что подобное переключение setSize() и setVisible() заставляет его работать. Пробовал это около дюжины раз без проблем.

Я думаю, setVisible может не понравиться фокусировка на окне, если оно имеет размер 0x0. Вопрос в том, почему это вызывает проблему только в одном из некоторых случаев?

Комментарии:

1. Используйте не KeyListener, а привязки клавиш . Тогда фокус не так уж важен.

2. Периодически возникающие проблемы всегда напоминают мне о начальных потоках .

Ответ №1:

Попробуйте добавить JButton в вашу JPanel «canvas», затем нажмите кнопку и посмотрите, что происходит с вашим KeyListener — это сбой, потому что JPanel потеряла фокус. Чтобы этого не произошло, используйте вместо этого привязки клавиш (см. Ссылку на руководство в моем комментарии выше). Например,,

 import java.awt.event.*;
import java.awt.*;
import javax.swing.*;

@SuppressWarnings("serial")
public class Game2 {

   private static final String UP = "up";

   public static void main(String[] args) {
      new Game2();
   }

   public Game2() {
      JFrame window = new JFrame("Press up-arrow key");
      window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

      JPanel canvas = new JPanel();
      canvas.setPreferredSize(new Dimension(400, 300));
      window.add(canvas);

      canvas.add(new JButton(new AbstractAction("Press space-bar") {
         public void actionPerformed(ActionEvent e) {
            System.out.println("Button or space-bar pressed");
         }
      }));
      ActionMap actionMap = canvas.getActionMap();
      int condition = JComponent.WHEN_IN_FOCUSED_WINDOW;
      InputMap inputMap = canvas.getInputMap(condition);

      inputMap.put(KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0), UP);
      actionMap.put(UP, new UpAction());

      window.pack();
      window.setLocationRelativeTo(null);
      window.setVisible(true);
   }
}

@SuppressWarnings("serial")
class UpAction extends AbstractAction {
   @Override
   public void actionPerformed(ActionEvent arg0) {
      System.out.println("Up Arrow pressed!");
   }
}
  

Комментарии:

1. Спасибо за пример кода. Я уже читаю вашу ссылку. Это оставляет вопрос о том, почему ошибка возникает так случайно.

Ответ №2:

Не знаю, связано ли это с вашими проблемами, но из-за прерывистого характера этого, возможно, так оно и есть…Вы должны выполнить setVisible() последним и в потоке swing. Вы могли бы вызвать setSize после setVisible, если хотите, но пользователь может увидеть мерцание, и это аналогично должно быть сделано в потоке swing. Сделайте это в качестве вашего последнего шага:

 SwingUtilities.invokeLater( new Runnable() {
   public void run() {
      window.setVisible( true );
   }
} );
  

Для этого вам также нужно будет сделать объявление window окончательным:

 ...
final JFrame window = new JFrame();
...
  

Комментарии:

1. Поскольку вызов setVisible () после первого setSize () решает проблему, это тоже должно сработать нормально. Я не вызывал его последним для начала, потому что я хотел получить вставки Frames, чтобы получить размер строки заголовка / границ вокруг нее. Это работает только при предварительном вызове setVisible().