我有一个列表,其中每个项目都有几个内容,包括JProgressBar
,可以进行大量更新。每当其中一个项目更新其JProgressBar
时,列表中的ListDataListener
会尝试使用
/*
* This makes the updating content item automatically scroll
* into view if it is off the viewport.
*/
public void contentsChanged(final ListDataEvent evt) {
if (!EventQueue.isDispatchThread()) {
/**
* Make sure the scrolling happens in the graphics "dispatch" thread.
*/
EventQueue.invokeLater(new Runnable() {
public void run() {
contentsChanged(evt);
}
});
}
if (playbackInProgress) {
int index = evt.getIndex0();
currentContentList.ensureIndexIsVisible(index);
}
}
请注意,我正在尝试确保在调度线程中完成滚动,因为我认为问题可能是它在重新绘制时滚动。然而,我仍然有一个问题,如果事情真的活跃,一些列表项在视口之外绘制,覆盖JScrollPane
之外的内容。强制曝光事件会重绘那些东西,但这很烦人。
还有什么我需要注意的,以阻止这些东西在剪裁区域之外绘画吗?
答案 0 :(得分:3)
您是否尝试在JList和/或它正在绘制的组件上显式启用双缓冲? (附:setDoubleBuffered(boolean aFlag)
)
另一个想法是,可能需要在委派给EDT后立即退出该功能。编写代码的方式,如果从非EDT线程调用ContentChanged
,则看起来更新将在两个线程中发生。登录第一个if
(或在if - 但是不是中设置一个断点 - 应该有助于确定这是否是您的问题。
例如:
public void contentsChanged(final ListDataEvent evt)
{
if (!EventQueue.isDispatchThread())
{
log.debug("Delegating contentsChanged(...) to EDT");
EventQueue.invokeLater(new Runnable()
{
public void run()
{
contentsChanged(evt);
}
});
// don't run ensureIndexIsVisible twice:
return;
}
if (playbackInProgress)
{
int index = evt.getIndex0();
currentContentList.ensureIndexIsVisible(index);
}
}