#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().