Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/java/351.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Java-从单独的组件中重新绘制组件_Java_Swing_Class_User Interface_Oop - Fatal编程技术网

Java-从单独的组件中重新绘制组件

Java-从单独的组件中重新绘制组件,java,swing,class,user-interface,oop,Java,Swing,Class,User Interface,Oop,我对Java中的GUI编程非常陌生,遇到了一个问题,即一个组件中的更改会导致另一个组件(不是父、子、祖父母或孙子等)必须重新绘制。更具体地说,我有一个窗口作为winGUI(JFrame)和一个BorderLayout contentPane的子类,其中的中心包含一个具有以下方法的组件,该方法在单击组件时运行: winGUI window = (winGUI)(getParent().getParent().getParent().getParent()); window.updatePanel(

我对Java中的GUI编程非常陌生,遇到了一个问题,即一个组件中的更改会导致另一个组件(不是父、子、祖父母或孙子等)必须重新绘制。更具体地说,我有一个窗口作为
winGUI
JFrame
)和一个BorderLayout contentPane的子类,其中的中心包含一个具有以下方法的组件,该方法在单击组件时运行:

winGUI window = (winGUI)(getParent().getParent().getParent().getParent());
window.updatePanel(panelContent);
(我想有更好的方法检索窗口,所以最好知道。)

winGUI的类定义如下

import java.awt.BorderLayout;
import java.awt.Toolkit;
import javax.swing.JFrame;
import javax.swing.JPanel;

public class winGUI extends JFrame {
    private UIController userInterface;
    private JPanel content;

    public static void main(String[] args) {
        JFrame window = new winGUI();
        window.setResizable(false);
        window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        window.setVisible(true);
    }

    public winGUI() {
        super("Window");
        setSize(Toolkit.getDefaultToolkit().getScreenSize());
        setResizable(false);
        content = new JPanel();
        content.setLayout(new BorderLayout());
        userInterface = new UIController();
        content.add(userInterface.getCenter(), BorderLayout.CENTER);
        content.add(userInterface.getPanel(), BorderLayout.WEST);
        setContentPane(content);
        setLocation(0, 0);
    }

    public void updatePanel(PanelContent panelContent) {
        userInterface.buildPanel(panelContent);
        content.add(userInterface.getPanel(), BorderLayout.WEST);
    }

}
以下是UIController类的重要部分:

private JPanel sidePanel;

public UIController() {
    buildPanel(null);
}

public void buildPanel(PanelContent panelContent) {
    sidePanel = new JPanel();
    sidePanel.setBackground(new Color(220, 220, 220));
    sidePanel.setLayout(new FlowLayout(FlowLayout.LEFT, 5, 5));
    sidePanel.setPreferredSize(new Dimension(250, 500));
    if (panelContent != null) {
        sidePanel.add(new JLabel(panelContent.getLabel()));
    }
}

public JPanel getPanel() {
    return sidePanel;
}
updatePanel
方法中调用
userInterface.buildPanel(panelContent)
后,我希望面板会更改,因为getPanel()方法通过引用传递,因此,创建窗口时添加到
content
的组件以及
sidePanel
都应该更新,因为它们引用相同的对象。然而,这并没有发生,所以我尝试添加

content.add(userInterface.getPanel(), BorderLayout.WEST);
userInterface.getPanel().repaint();
this.repaint();
updatePanel
方法的末尾,但无效

非常感谢您的帮助。

您可以使用(创建自己的类似Swing的侦听器)将事件传递给
winGui
,它应该更新其面板

interface SomeUpdateListener {
     public void onUpdate(PanelContent panelContent);
}
然后让winGui成为这样一个倾听者:

public class winGUI extends JFrame implements SomeUpdateListener
    ...
    @Override
    public void onUpdate(PanelContent panelContent) {
        userInterface.buildPanel(panelContent);
        content.add(userInterface.getPanel(), BorderLayout.WEST);
        // Fix for your last issue I believe
        this.revalidate();  // Pretty sure this is what's needed in your case
        this.repaint(); // redraws them
    }
我不知道定制组件调用updatePanel实际使用的是什么,也不知道它是如何工作的,但我们假设它被称为
SomeCustomComponent
,现在大致有以下代码:

public class SomeCustomComponent extends JComponent {
    private SomeUpdateListener listener;
    public SomeCustomComponent(SomeUpdateListener someListener) {
         this.listener = someListener;
    }
    ...
    private void timeToMakeThatPanelContentUpdate() {
         PanelContent newContent = createNewPanelContents();
         listener.onUpdate(newContent);
    }
    ...
因此,现在使用这种新的“侦听器”类型,您的父级(或者我这里假设的最终父级…)和应用程序代码中的子组件之间没有直接的循环引用

注意:我添加了几行,以修复面板的最后一个问题,即在
onUpdate()
方法中没有使用最新内容进行更新

this.revalidate(); // Pretty sure this is what's needed in your case
this.repaint();

此外,如果您正在创建的面板在某些GUI组件组合中实际上没有任何不同,而只是与它显示的信息不同,那么您可能应该只传递表示先前存在的GUI组件现在应该显示的内容的数据。

“我想有更好的方法检索窗口”——但我怀疑这一点的使用。更好的解决方案是通过组件层次结构向下传递某种管理器/控制器,该层次结构描述了允许子组件执行的操作。您可能需要使组件层次结构无效,以便强制它更新其布局。尝试调用
revalidate
然后调用
repaint
我尝试使用revalidate,结果有些奇怪。(我原来问题中的代码被简化了;在我的实际程序中,侧面板有
JPanel
s包含
JLabel
s、
JButton
s和
JTextField
s)一个
JTextField
是我看到的唯一绘制的东西,但按钮是在我将鼠标悬停在应该绘制的位置时绘制的(当我移开鼠标时,它会保持不变)如果我单击它应该在的位置,大多数其他的
JTextField
就会出现。
JLabel
仍然没有出现。它看起来像swing中的一些bug,但我不认为这会被遗漏,因为我没有做任何不寻常的事情。不,这不是swing中的bug,它是线程问题、空布局问题或是面板已损坏的事实没有意识到这一点。如果没有一个a来说明你的问题,那就是猜测。注意:这不是一个代码转储,而是一个你正在做的事情的例子,它突出了你的问题。这将减少混乱,提高效率responses@MadProgrammer我现在已经修好了;我必须从
conte在调用
buildPanel
、将更新的面板添加到
content
并调用
validate()
revalidate()
repaint()之前,nt
位于西部
。谢谢你的建议。谢谢你的回答,我本来会将window对象传递给构造函数的,但它已经有4个参数,并且在我继续处理它时可能会获得更多参数。重新验证是我缺少的关键元素之一,另一个是在重新生成并添加它之前。很抱歉,我无法对你的答案进行投票;我不知道“没有足够的分数//\(-u-)/\@Zyxl没问题。你可以(我认为)通过进入帮助->浏览并滚动页面到底部,获得大部分初始分数。这应该足够让你至少可以对人们的问题添加评论。”。