如何防止JColorChooser从JComponent上吸收keyeevents ?



我的问题是。我有一个JComponent在滑动条的中间,它本身在边框的BorderLayout的中间。我用我的JComponent在里面画/写东西。

我的问题来了,当我在边界布局的一侧实现了一个JColorChooser。它应该可以改变图纸的颜色。一旦我在JColorChooser上选择了一种颜色,我的JComponent就停止响应KeyEvents,我就不能再写任何东西了。

我的代码简化如下:

My main:

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没有响应keyeevents。我希望我把我的问题说清楚了,提前谢谢你

在帧中使用KeyListener的小操作是因为JComponent不响应keyeevents。

不需要。不要在框架中添加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()。适配器类的默认实现不做任何事情。

这是因为你的组件失去了焦点,所以它将不再接收鼠标事件。我将鼠标侦听器更改为包含requestFocusInWindow,这是JComponent方法。焦点会回到你的PhotoComponent,你可以在点击面板后再次输入。

this.addMouseListener(new MouseAdapter() {
@Override
public void mouseClicked(MouseEvent e) {
super.mouseClicked(e);
currentText = new Text(e.getPoint());
textList.add(currentText);
requestFocusInWindow();
}
});