gpt4 book ai didi

java - 使用(重新)验证、重绘后 Swing ContentPane 不更新

转载 作者:行者123 更新时间:2023-11-30 07:21:01 25 4
gpt4 key购买 nike

类似的问题已经问过好几次了。参见例如herehere .

但我真的很想了解为什么我的代码不起作用。正如在这个问题的其他版本中所回答的那样,CardLayout 可能就足够了,但在我的情况下我不确定它是否理想。无论如何,我感兴趣的是从概念上理解为什么这行不通。

我有一个 JFrame,它的内容 Pane 监听关键事件。在内容 Pane 中按下某个键时,内容 Pane 会通知 JFrame 使用新的内容 Pane 更新自身。这是问题的一个简单示例:

此代码是完全可编译的。您可以复制粘贴它并按原样运行它。

这是我的 JFrame:

import javax.swing.*;
import java.awt.*;
import java.util.Random;

public class SimpleSim extends JFrame{

private static SimpleSim instance = null;

public static SimpleSim getInstance(){
if(instance == null){
instance = new SimpleSim();
}

return instance;
}

private SimpleSim(){}

public void initialize(){

this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
this.setExtendedState(Frame.MAXIMIZED_BOTH);
this.pack();
this.setVisible(true);

update();
}

public void update(){
System.out.println("SIMPLE_SIM UPDATE THREAD: " + Thread.currentThread().getName());
Random rand = new Random();

float r = rand.nextFloat();
float g = rand.nextFloat();
float b = rand.nextFloat();

SimplePanel simplePanel = new SimplePanel(new Color(r, g, b));
JPanel contentPane = (JPanel) this.getContentPane();

contentPane.removeAll();
contentPane.add(simplePanel);
contentPane.revalidate();
contentPane.repaint();

validate();
repaint();

}


}

这是用作我的内容 Pane 的 JPanel:

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


public class SimplePanel extends JPanel implements KeyListener {

public SimplePanel(Color c){

setFocusable(true);
setLayout(null);
setBackground(c);
setVisible(true);
this.addKeyListener(this);
}
public void keyTyped(KeyEvent keyEvent) {
if(keyEvent.getKeyChar() == 'a'){
System.out.println("a");
System.out.println("SIMPLE_PANEL KEY PRESS THREAD: " + Thread.currentThread().getName());
SimpleSim.getInstance().update();
}
}

public void keyPressed(KeyEvent keyEvent) {
}
public void keyReleased(KeyEvent keyEvent) {
}
}

奇怪的是,它在您第一次按 a 时有效,但之后就无效了。我的猜测是这里存在线程问题。我可以看到当第一次调用 update 时,它是在主线程上调用的。下次在美国东部时间调用它。我试过使用 invokeLater() 调用 update() ,但也没有用。我找到了一种使用不同设计模式的解决方法,但我真的很想了解为什么这不起作用。

此外,要运行的简单类:

public class Run {

public static void main(String[] args){
SimpleSim.getInstance().initialize();
}
}

注意:验证和重新绘制 JFrame 的看似多余的调用是为了尝试安抚发布在我提供的第二个链接上的建议,该建议指出:在受影响最大的组件上调用 validate()。这可能是 Java 渲染周期中最困惑的部分。对 invalidate 的调用将组件及其所有祖先标记为需要布局。对验证的调用执行组件及其所有后代的布局。一个“向上”工作,另一个“向下”工作。您需要在树中将受您的更改影响的最高组件上调用验证。我认为这会使它起作用,但无济于事。

最佳答案

抱歉,我对您的代码做了一些修改,但这让测试变得非常容易......

我能看到的导入更改是在更新方法中。基本上我只是在框架上调用了revalidate

重新验证状态

Revalidates the component hierarchy up to the nearest validate root.

This method first invalidates the component hierarchy starting from this component up to the nearest validate root. Afterwards, the component hierarchy is validated starting from the nearest validate root.

This is a convenience method supposed to help application developers avoid looking for validate roots manually. Basically, it's equivalent to first calling the invalidate() method on this component, and then calling the validate() method on the nearest validate root.

我认为最后一部分是您自己的代码中缺少的内容。

public class SimpleSim extends JFrame {

private static SimpleSim instance = null;

public static SimpleSim getInstance() {
if (instance == null) {
instance = new SimpleSim();
}

return instance;
}

private SimpleSim() {
}

public void initialize() {

this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
this.setSize(400, 400);
this.setVisible(true);
setLayout(new BorderLayout());

update();

}

public void update() {
System.out.println("NEXT: " + Thread.currentThread().getName());
Random rand = new Random();

float r = rand.nextFloat();
float g = rand.nextFloat();
float b = rand.nextFloat();

SimplePanel simplePanel = new SimplePanel(new Color(r, g, b));
JPanel contentPane = (JPanel) this.getContentPane();

getContentPane().removeAll();
add(simplePanel);

revalidate();
}

public class SimplePanel extends JPanel {

public SimplePanel(Color c) {

setFocusable(true);
setLayout(null);
setBackground(c);
setVisible(true);
requestFocusInWindow();
InputMap im = getInputMap(WHEN_IN_FOCUSED_WINDOW);
ActionMap am = getActionMap();

im.put(KeyStroke.getKeyStroke(KeyEvent.VK_A, 0), "A");
am.put("A", new AbstractAction() {
@Override
public void actionPerformed(ActionEvent e) {
System.out.println("a");
System.out.println("KEY: " + Thread.currentThread().getName());
SimpleSim.getInstance().update();
}
});

}
}

public static void main(String[] args) {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
}

SimpleSim.getInstance().initialize();
}
});
}
}

此外,我建议您利用 key bindings API 而不是使用 KeyListener。它将解决一些焦点问题;)

更新

经过一段时间测试各种排列后,我们得出结论,主要问题与焦点问题有关。

虽然 SimplePanel 是可聚焦的,但没有任何东西为其提供焦点,这意味着关键监听器无法被触发。

添加 simplePanel.requestFocusInWindow 在它被添加到框架之后似乎允许关键监听器保持 Activity 状态。

根据我自己的测试,没有调用 revalidate 面板没有更新。

关于java - 使用(重新)验证、重绘后 Swing ContentPane 不更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13576985/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com