【问题标题】:Have something loaded only when JList item is visibile仅在列表项可见时才加载某些内容
【发布时间】:2010-12-12 08:50:51
【问题描述】:

我正在实现一个包含很多元素的 Jlist。每个元素对应一个图像,所以我想在列表的每一行中显示调整大小的预览。我已经实现了一个自定义 ImageCellRenderer 扩展 Jlabel 并在 getListCellRendererComponent 上创建缩略图(如果该元素没有任何缩略图)。每行对应一个 Page 类,我在其中存储图像的路径和应用于 JLabel 的图标。每个 Page 对象都放在 DefaultListModel 中以填充 JList。 渲染代码是这样的:

        public Component getListCellRendererComponent(
            JList list,
            Object value,
            int index,
            boolean isSelected,
            boolean cellHasFocus)
    {
        Page page = (Page) value;

        if (page.getImgIcon() == null)
        {
            System.out.println(String.format("Creating thumbnail of %s", page.getImgFilename()));
            ImageIcon icon = new ImageIcon(page.getImgFilename());

            int thumb_width = icon.getIconWidth() > icon.getIconHeight() ? 128 : ((icon.getIconWidth() * 128) / icon.getIconHeight());
            int thumb_height = icon.getIconHeight() > icon.getIconWidth() ? 128 : ((icon.getIconHeight() * 128) / icon.getIconWidth());
            icon.setImage(getScaledImage(icon.getImage(), thumb_width, thumb_height));

            page.setImgIcon(icon);
        }

        setIcon(page.getImgIcon());

    }

我在想只有某个项目在调用单元格渲染器的列表中是可见的,但是当我将 Page 对象添加到列表模型时,我看到所有缩略图都是创建的。我尝试加载项目,然后在 JList 中设置模型或先设置模型,然后开始附加项目,但结果是相同的。 有什么方法可以仅在必要时加载数据,还是我需要创建一个自定义控件,如 JScrollPanel,里面有堆叠的项目,我可以在其中检查每个元素的可见性?

谢谢

【问题讨论】:

    标签: java visibility jlist cellrenderer


    【解决方案1】:

    我相信 JList 会遍历所有项目并调用渲染器以确定列表的首选大小。

    您可以通过使用 setPrototypeCellValue(...) 方法来防止这种情况。当然,渲染器仍然会为滚动窗格中可见的所有项目调用。

    【讨论】:

    • 我首先为所有项目分配了一个固定大小的空图像,因此第一个渲染计算列表的总高度。在项目的第二次绘制请求中,我加载了元素的真实图像。谢谢
    【解决方案2】:

    罪魁祸首似乎是其updateLayoutState() 中的BasicListUI。在绘制操作期间,它会在设置 updateLayoutStateNeeded 标志时遍历 dataModel 中的每个元素(并且在看似每次属性更改之后设置,以及从 @987654325 添加和删除间隔时) @)。在每个单元格上,它将调用getListCellRendererComponent 方法并导致您加载图标。如果您在 JList 实例上设置 fixedCellHeightfixedCellWidth 字段,则可以阻止这种情况发生。

    您可以尝试使用两个公共方法设置固定的高度和宽度:

    • setFixedCellHeight(int)
    • setFixedCellWidth(int)

    顺便说一句:一般来说,如果在事件调度线程中执行,从文件系统加载图标可能会很慢并导致您的 GUI 感觉迟缓。这通常是大图像的更多问题,其中加载文件可能需要时间。如果您还没有,您可能想要考虑在单独的线程中加载图标。这可以通过预先加载一个共享的“图标不可用图像”来完成,并且当您向Page 查询图标时,如果特定于页面的图标为空,您可以触发所需文件的加载,然后返回“图标不可用图像”,一旦加载了Page,将其分配给Page 实例并触发重绘。只是一个想法。

    【讨论】:

    • 我创建了一个图标对象,其中包含一个虚拟图像,如您所说,它作为未加载的预览。在第一次调用 getListCellRendererComponent 时,我用这个对象设置了页面图标。通过这种方式,列表填充了正确的滚动尺寸。在列表元素绘制的第二次调用中,我使用 SwingWorker 线程加载调整大小的图像。通过这种方式,我可以滚动列表而不会减慢速度,但是在加载线程完成后我无法强制重新绘制图标(我看到了虚拟图像)。仅调整窗口大小会更新图像,并且 JLabel 的任何重绘都不起作用。
    • 您需要做的是触发您的 ListModel 触发 ListDataEvent 到它的 ListDataListeners,这将指示 UI 更新其视图。如果你有子类AbstractListModel,它有一个受保护的fireContentsChanged 方法应该可以为你解决问题。请记住,这应该在 EDT 内运行。
    猜你喜欢
    • 1970-01-01
    • 2013-04-06
    • 1970-01-01
    • 1970-01-01
    • 2013-09-13
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多