這是我的問題。我在滑塊中間有一個 JComponent,它本身在框架中的 BorderLayout 中間。我使用我的 JComponent 在其中繪制/寫入內容。
當我在邊框布局的一側實作 JColorChooser 時,我的問題就出現了。它應該改變圖紙的顏色。一旦我在 JColorChooser 上選擇了一種顏色,我的 JComponent 就停止回應 KeyEvents 并且我不能再寫任何東西了。
這里我的代碼簡化了:
我的主要:
import javax.swing.*;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import java.awt.*;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import java.awt.image.BufferedImage;
public class TestPhotoComponent {
public static void main(String[] args) {
JFrame frame = new JFrame("PhotoComponent test");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new BorderLayout());
JPanel p1 = new JPanel();
p1.setPreferredSize(new Dimension(Integer.MAX_VALUE, 100));
p1.setBackground(Color.GREEN);
frame.add(p1, BorderLayout.NORTH);
JPanel p2 = new JPanel();
p2.setPreferredSize(new Dimension(Integer.MAX_VALUE, 100));
p2.setBackground(Color.GREEN);
frame.add(p2, BorderLayout.SOUTH);
JPanel p3 = new JPanel();
p3.setPreferredSize(new Dimension(100, Integer.MAX_VALUE));
p3.setBackground(Color.GREEN);
frame.add(p3, BorderLayout.WEST);
JPanel p4 = new JPanel();
p4.setPreferredSize(new Dimension(200, Integer.MAX_VALUE));
p4.setBackground(Color.GREEN);
frame.add(p4, BorderLayout.EAST);
PhotoComponent pc = new PhotoComponent(new Dimension(500, 300));
pc.setFocusable(true);
JScrollPane sp = new JScrollPane(pc);
sp.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
sp.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
sp.getViewport().setBackground(Color.CYAN);
frame.add(sp, BorderLayout.CENTER);
final JColorChooser colorChooser = new JColorChooser();
colorChooser.getSelectionModel().addChangeListener(new ChangeListener() {
@Override
public void stateChanged(ChangeEvent e) {
Color color = colorChooser.getColor();
if (color != null)
pc.setColor(color);
}
});
p4.add(colorChooser);
frame.setFocusable(true);
frame.setPreferredSize(new Dimension(900, 600));
frame.pack();
frame.setVisible(true);
frame.addKeyListener(new KeyAdapter() {
@Override
public void keyTyped(KeyEvent e) {
pc.keyListener.keyTyped(e);
}
});
}
}
我的 JComponent :
import javax.swing.*;
import java.awt.*;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.util.ArrayList;
public class PhotoComponent extends JComponent {
Point position = null;
ArrayList<Text> textList = new ArrayList<>();
Text currentText = null;
Color color = Color.BLACK;
KeyAdapter keyListener;
public PhotoComponent(Dimension dimension){
this.setFocusable(true);
this.setVisible(true);
this.setPreferredSize(dimension);
addListeners();
}
void addListeners(){
this.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
super.mouseClicked(e);
currentText = new Text(e.getPoint());
textList.add(currentText);
}
});
this.keyListener = new KeyAdapter() {
@Override
public void keyTyped(KeyEvent e) {
super.keyTyped(e);
if(currentText != null){
currentText.addCharacter(e.getKeyChar());
repaint();
}
}
};
this.addKeyListener(this.keyListener);
}
@Override
public void paintComponent(Graphics g)
{
super.paintComponent(g);
Graphics2D g2 = (Graphics2D) g;
Color oldColor = g2.getColor();
g2.setColor(Color.WHITE);
g2.fill(this.getBounds());
g2.setColor(this.color);
for(Text t : textList){
t.draw(g2);
}
g2.setColor(oldColor);
}
public void setColor(Color color) {
this.color = color;
}
}
還有我的 Text 物件
import java.awt.*;
public class Text {
Point position = null;
String content = "";
public Text(Point position){
this.position = position;
}
public void addCharacter(char newCharacter){
this.content = newCharacter;
}
public void draw(Graphics2D g2){
g2.drawString(content, position.x, position.y);
}
}
對框架中的 KeyListener 進行的少量操作是因為 JComponent 不回應 KeyEvents。我希望我已經解決了我的問題,并提前致謝
uj5u.com熱心網友回復:
這是因為您的組件失去了焦點,因此它將不再接收滑鼠事件。我更改了您的滑鼠偵聽器以包含requestFocusInWindowJComponent 方法。焦點將回傳到您的 PhotoComponent,您可以在單擊面板后再次鍵入。
this.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
super.mouseClicked(e);
currentText = new Text(e.getPoint());
textList.add(currentText);
requestFocusInWindow();
}
});
uj5u.com熱心網友回復:
對框架中的 KeyListener 進行的少量操作是因為 JComponent 不回應 KeyEvents。
這是不需要的。不要將 KeyListener 添加到框架中。
僅將 KeyListener 添加到 PhotoComponent。
KeyEvent 僅分派給具有焦點的組件。因此,您希望 PhotoComponent 在框架可見時獲得焦點。
您可以通過添加以下內容來做到這一點:
/*
frame.addKeyListener(new KeyAdapter() {
@Override
public void keyTyped(KeyEvent e) {
pc.keyListener.keyTyped(e);
}
});
*/
pc.requestFocusInWindow();
我不明白為什么它在 changeListener 中不起作用
有時,Swing 組件會向 EDT 添加代碼,因此代碼不會按照您期望的順序執行。因此 PhotoComponent 可能會獲得焦點,但顏色選擇器可能會將其奪回。
因此,在您的 ChangeListener 中,您首先嘗試:
pc.requestFocusInWindow();
在這種情況下,它不起作用。
因此,更改偵聽器中的邏輯應該是:
public void stateChanged(ChangeEvent e) {
Color color = colorChooser.getColor();
if (color != null)
pc.setColor(color);
SwingUtilities.invokeLater(() -> pc.requestFocusInWindow());
}
這將確保在所有顏色選擇器邏輯執行完畢后執行代碼。
你 setColor() 方法也應該是:
this.color = color;
repaint();
當組件的屬性發生更改時,Swing 組件應該負責重新繪制自身,就像在更改文本時重新繪制組件一樣。
同樣在撰寫偵聽器時,則無需呼叫super()該事件。配接器類的默認實作什么都不做。
轉載請註明出處,本文鏈接:https://www.uj5u.com/yidong/333801.html
上一篇:在CSS/Styled組件中,將絕對位置置于頂部中心位置
下一篇:C++檔案操作fstream
