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

Posted

技术标签:

【中文标题】Java - 从单独的组件中重绘组件【英文标题】:Java - Redrawing a component from within a separate component 【发布时间】:2015-04-21 13:31:53 【问题描述】:

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

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方法结束,但无济于事。

非常感谢任何帮助。

【问题讨论】:

“我怀疑有更好的方法来检索窗口” - SwingUtilities.windowForComponent - 但我会质疑它的用途。更好的解决方案是通过组件层次结构向下传递某种管理器/控制器,该层次结构描述了允许哪些子组件执行的操作 您可能需要使组件层次结构无效以强制它更新其布局。尝试拨打revalidate,然后拨打repaint 我尝试使用 revalidate 并得到了一些奇怪的结果。 (我最初的问题中的代码被简化了;在我的实际程序中,侧面板有JPanels,其中包含JLabels、JButtons 和JTextFields。)一个JTextField 是我看到的唯一绘制的东西,但是当我将鼠标悬停在它们应该在的位置时会绘制按钮(当我将鼠标移开时会保留),如果我单击它应该在的位置,大多数其他JTextField 会出现。 JLabels 仍然没有出现。它看起来像是挥杆中的一些错误,但我看不出这是怎么被错过的,因为我没有做任何不寻常的事情。 不,这不是 Swing 中的错误,它要么是线程问题,要么是空布局问题,或者是面板没有识别出需要布局的事实。如果没有 runnable example 来证明您的问题,这一切都是猜测。注意:这不是代码转储,而是您正在做的一个示例,它突出了您遇到的问题。这将导致更少的混乱和更好的响应 @MadProgrammer 我现在已经修好了;在调用buildPanel 之前,我必须从content 的WEST 到remove the old panel,将更新后的面板添加到content 并调用validate()revalidate()repaint()。感谢您的建议。 【参考方案1】:

您可以使用 observer pattern(创建您自己的 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 组件现在应该显示的数据。

【讨论】:

感谢您的回答,我会将 window 对象传递给构造函数,但它已经有 4 个参数,并且在我继续处理它时可能会获得更多参数。重新验证是我缺少的关键元素之一,另一个是 removing the old panel 在重建并添加它之前。抱歉,我无法对您的答案进行投票;我的积分不够 //\(-_-)//\\ @Zyxl 没问题。通过转到帮助 -> 游览并滚动页面到底部,您始终可以(我认为)获得大部分初始点。这应该足以让您至少在人们的问题上添加 cmets。

以上是关于Java - 从单独的组件中重绘组件的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Java 中使用 JButton 重绘组件?

来自线程内的实例化组件不会重绘到 Java 中的 JFrame

在 ExtJs4 中重命名组件的路径

重绘 Swing 组件会产生糟糕的结果

React组件性能优化总结

AgGrid RowData在Angular组件外部更新,需要重绘行