我试图使按钮在按下时改变颜色。我创建了以下代码:
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.EventQueue;
import java.awt.Font;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyEvent;
import javax.swing.AbstractAction;
import javax.swing.ActionMap;
import javax.swing.InputMap;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTextArea;
import javax.swing.KeyStroke;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.border.EmptyBorder;
public class KeyboardTest extends JFrame {
public static void main(String[] args) {
new KeyboardTest();
}
public KeyboardTest() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel("com.sun.java.swing.plaf.nimbus.NimbusLookAndFeel");
} catch (Exception ex) {
try {
UIManager.setLookAndFeel(UIManager
.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException e) {
e.printStackTrace();
}
}
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new BorderLayout());
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
frame.setBounds(100, 100, 650, 390);
}
});
}
public class TestPane extends JPanel {
public TestPane() {
JPanel contentPane = new JPanel();
contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
add(contentPane);
contentPane.setLayout(null);
JPanel keyboardPanel = new JPanel();
keyboardPanel.setBounds(10, 122, 614, 181);
contentPane.add(keyboardPanel);
keyboardPanel.setLayout(new GridLayout(5, 0, 0, 0));
... adding panels and buttons...
addKeyBinding(A, "A", KeyEvent.VK_A);
addKeyBinding(B, "B", KeyEvent.VK_B);
addKeyBinding(C, "C", KeyEvent.VK_C);
addKeyBinding(D, "D", KeyEvent.VK_D);
addKeyBinding(E, "E", KeyEvent.VK_E);
}
protected void addKeyBinding(JButton btn, String name, int virtualKey) {
ActionMap am = getActionMap();
InputMap im = getInputMap(WHEN_IN_FOCUSED_WINDOW);
im.put(KeyStroke.getKeyStroke(virtualKey, 0, false), name
+ ".pressed");
im.put(KeyStroke.getKeyStroke(virtualKey, 0, true), name
+ ".released");
am.put(name + ".pressed", new KeyAction(btn, true));
am.put(name + ".released", new KeyAction(btn, false));
}
}
public class KeyAction extends AbstractAction {
private JButton btn;
private boolean highlight;
public KeyAction(JButton btn, boolean highlight) {
this.btn = btn;
this.highlight = highlight;
}
@Override
public void actionPerformed(ActionEvent e) {
if (highlight) {
btn.getModel().setPressed(true);
btn.setBackground(Color.RED);
btn.setOpaque(true);
} else {
btn.getModel().setPressed(false);
btn.setBackground(null);
btn.setOpaque(false);
}
}
}
}
这是应该的样子:
这是实际显示的内容:
我应该附上完整的代码吗? (很长...)
最佳答案
无法测试您的代码(由于缺少组件)。我不会对您介绍null布局。但是,这是一个问题,并且根据我能够测试的内容(使用自己的组件)解决方案。
我看到一个没用的JPanel contentPane
。您正在使用的类已经是一个面板,我看不出创建另一个然后将其包装到另一个中的意义。 (虽然不是整个问题)
TestPane类具有默认的FlowLayout,这意味着它将尊重内部组件的首选大小,即contentPane
面板。问题在于它没有首选的大小。您可以使用setBounds
设置按钮大小。这与preferredSize
不同。因此,TestPane
面板没有可计算的首选大小。
简单的解决方法是将TestPane
的布局设置为BorderLayout
。这不会考虑首选大小,并且会拉伸contentPane,以便您可以看到它。但是,与此相关的问题是您需要设置框架的大小。 pack()无法正常工作,因为整个首选大小交易。这就是为什么帧在第一次打开时会闪烁的原因,因为它会先缩小,然后再使用setBounds进行扩展。
就像我在评论中说的那样,我会避免使用null布局(出于许多原因)。您可以从疯子本人@MadProgrammer看到一个使用GridBagLayout here as posted in my comment的键盘的好例子。