Java - 从单独的组件中重绘组件
Posted
技术标签:
【中文标题】Java - 从单独的组件中重绘组件【英文标题】:Java - Redrawing a component from within a separate component 【发布时间】:2015-04-21 13:31:53 【问题描述】:我对 Java 中的 GUI 编程非常陌生,并且遇到了一个问题,即一个组件的更改导致另一个组件(不是父或子或祖父母或孙子等)必须重新绘制。更具体地说,我有一个带有 BorderLayout contentPane 的 winGUI
(JFrame
的子类)的窗口,其中的 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 并得到了一些奇怪的结果。 (我最初的问题中的代码被简化了;在我的实际程序中,侧面板有JPanel
s,其中包含JLabel
s、JButton
s 和JTextField
s。)一个JTextField
是我看到的唯一绘制的东西,但是当我将鼠标悬停在它们应该在的位置时会绘制按钮(当我将鼠标移开时会保留),如果我单击它应该在的位置,大多数其他JTextField
会出现。 JLabel
s 仍然没有出现。它看起来像是挥杆中的一些错误,但我看不出这是怎么被错过的,因为我没有做任何不寻常的事情。
不,这不是 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 - 从单独的组件中重绘组件的主要内容,如果未能解决你的问题,请参考以下文章