为什么SWT Composite有时需要调用resize()才能正确布局?

有时候我们会遇到一个绝对拒绝正确摆脱的SWT复合材料。 当我们在复合材料上调用dispose然后用另一个替换它时,我们经常会遇到这种情况; 虽然它似乎并不严格限于这种情况。

当我们遇到这个问题时,大约50%的时间,我们可以在有问题的复合材料上调用pack()layout() ,一切都会好的。 但是,大约50%的时间我们必须这样做:

 Point p = c.getSize(); c.setSize(p.x+1, p.y+1); c.setSize(p); 

几乎每种布局管理器组合都会发生这种情况。

我希望我有一个漂亮,简单,可重复的案例,但我没有。 我希望有人会认出这个问题并说:“好吧,呃,你错过了xyz ……”

在我看来, 布局的缓存已经过时,需要刷新

SWT中的布局支持缓存,并且通常会缓存控件的首选大小,或者他们喜欢缓存的任何内容:

 public abstract class Layout { protected abstract Point computeSize (Composite composite, int wHint, int hHint, boolean flushCache); protected boolean flushCache (Control control) {...} protected abstract void layout (Composite composite, boolean flushCache); } 

我对SWT编程(以前的Swing程序员)比较陌生,但遇到了布局没有正确更新的类似情况。 我通常能够使用其他布局方法解决它们,这也会导致布局刷新其缓存:

 layout(boolean changed) layout(boolean changed, boolean allChildren) 

希望有帮助……

与此同时,我在运行时更改或调整控件层次结构的某些部分时,更多地了解了SWT的缺点。 当应该调整其最小或首选内容大小时,还需要显式更新ScrolledCompositeExpandBar

我写了一个小帮助方法,为已更改的控件重新validation控件层次结构的布局:

 public static void revalidateLayout (Control control) { Control c = control; do { if (c instanceof ExpandBar) { ExpandBar expandBar = (ExpandBar) c; for (ExpandItem expandItem : expandBar.getItems()) { expandItem .setHeight(expandItem.getControl().computeSize(expandBar.getSize().x, SWT.DEFAULT, true).y); } } c = c.getParent(); } while (c != null && c.getParent() != null && !(c instanceof ScrolledComposite)); if (c instanceof ScrolledComposite) { ScrolledComposite scrolledComposite = (ScrolledComposite) c; if (scrolledComposite.getExpandHorizontal() || scrolledComposite.getExpandVertical()) { scrolledComposite .setMinSize(scrolledComposite.getContent().computeSize(SWT.DEFAULT, SWT.DEFAULT, true)); } else { scrolledComposite.getContent().pack(true); } } if (c instanceof Composite) { Composite composite = (Composite) c; composite.layout(true, true); } } 

复合材料的布局负责布置该复合材料的子元素。 因此,如果复合材料的大小没有变化,但需要更新子项的相对位置和大小,则可以在复合材料上调用layout() 。 但是,如果需要更新组合本身的大小或位置,则必须在其父组合上调用layout() (依此类推,直到到达shell)。

经验法则:如果您添加或删除了控件,或以其他方式完成了需要重新布局的操作,请向上移动窗口小部件层次结构,直到找到带有滚动条的复合并在其上调用layout() 。 使用滚动条停止复合的原因是它的大小不会随着变化而改变 – 它的滚动条会“吸收”它。

请注意,如果需要布局的更改不是新子项或已删除的子项,则应在调用布局之前调用Composite.changed(new Control[] {changedControl})

我刚刚意识到Composite.changed(Control [] children)。 几年前我读到了一篇文章:

http://www.eclipse.org/articles/article.php?file=Article-Understanding-Layouts/index.html

本文还提到调用Composite.layout(boolean changed,boolean all)来更新布局:“调用layout()与调用布局(true)相同,它告诉ColumnLayout在设置子节点之前刷新其缓存“。 这一切都是正确的,从那时起我一直在做的事情。 但它并不是人们想要的,因为当你想要更新布局时,它基本上会失去布局缓存的好处,因为一个或几个控件已经改变了要求。

想象一下,GridLayout中有一堆StyledText小部件,您需要更改其中一个小部件的大小。 在StyledText上调用computeSize()非常昂贵。 而不是这个:

错误:

 parent.layout(true); 

…对所有孩子调用computeSize(),即使他们的要求没有改变。 你应该做这个:

对:

 parent.changed(new Control[] { theChangedChild }); 

然后

 rootComposite.layout(false, true); 

要么

 parent.layout(false); 

不是很直观。 layout()的参数命名错误。 它不应该称之为“已更改”,而应将其称为“ignoreCache”或其他内容。 直观的是当事情发生变化时传递“真实”。 相反,您需要传递“false”,但在执行之前,仅使用changed()更改控件的缓存无效…

请注意,调用changed()也会递归地使其父级中的父控件的缓存无效,这完全有意义。 因此,当您调用layout()时,应该使用all = true在根复合(通常是Shell)上调用它,除非您知道更改的控件的父级的大小将会响应或不能更改。