【问题标题】:JTree entry rendering issueJTree 入口渲染问题
【发布时间】:2015-11-06 09:33:05
【问题描述】:

在我的项目的 GUI 中,我使用 JTree 来显示文件系统,我遇到了一个特定于渲染的问题:似乎上述 JTree 中的某些条目能够正确渲染以及渲染它们自己之前选择的条目。这很难解释,所以我认为附上屏幕截图最容易理解。

为此,我使用箭头键从上到下快速移动。

我不太清楚这个问题的根源,但我怀疑这可能是我的TreeModelCellRenderer 的问题,它们都包含在下面。

欢迎任何可以帮助我解决此问题的帮助!

import java.awt.Component;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Arrays;
import java.util.Comparator;
import java.util.Vector;

import javax.imageio.ImageIO;
import javax.swing.Icon;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTree;
import javax.swing.WindowConstants;
import javax.swing.event.TreeModelListener;
import javax.swing.tree.DefaultTreeCellRenderer;
import javax.swing.tree.TreeModel;
import javax.swing.tree.TreePath;

public class MCVE {
    public static void main(String[] args) throws Throwable {
        JFrame frame = new JFrame();
        frame.getContentPane().add(new FileSystemTree(new File(System.getProperty("user.home"))));
        frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        frame.pack();
        frame.setVisible(true);
    }
}

class FileSystemTree extends JPanel {
    private File                root;
    private FileSystemTreeModel model;
    private final JTree         fileSystem;

    public FileSystemTree(File root) throws IOException {
        this.root = root;
        model = new FileSystemTreeModel(root);
        fileSystem = new JTree(model);
        fileSystem.setEditable(false);
        fileSystem.setCellRenderer(new FileSystemTreeCellRenderer());
        fileSystem.putClientProperty("JTree.lineStyle", "None");

        add(new JScrollPane(fileSystem));
    }
}

class FileSystemTreeCellRenderer extends DefaultTreeCellRenderer {
    private final Icon  folderSpecial;
    private final Icon  folderOther;
    private final Icon  fileSpecial;
    private final Icon  fileOther;

    public FileSystemTreeCellRenderer() throws IOException {
        InputStream stream;
        BufferedImage image;

        stream = new FileInputStream("folder-special.png");
        image = ImageIO.read(stream);
        folderSpecial = new ImageIcon(image);
        stream.close();

        stream = new FileInputStream("folder-other.png");
        image = ImageIO.read(stream);
        folderOther = new ImageIcon(image);
        stream.close();

        stream = new FileInputStream("file-special.png");
        image = ImageIO.read(stream);
        fileSpecial = new ImageIcon(image);
        stream.close();

        stream = new FileInputStream("file-other.png");
        image = ImageIO.read(stream);
        fileOther = new ImageIcon(image);
        stream.close();
    }

    @Override
    public Component getTreeCellRendererComponent(JTree tree, Object value, boolean sel, boolean expanded, boolean leaf, int row, boolean hasFocus) {
        super.getTreeCellRendererComponent(tree, value, sel, expanded, leaf, row, hasFocus);

        if (value instanceof File) {
            Icon icon = null;
            File file = (File) value;

            if (file.isDirectory()) {
                if (file.getName().endsWith("-special")) {
                    icon = folderSpecial;
                } else {
                    icon = folderOther;
                }
            } else if (file.isFile()) {
                if (file.getName().endsWith(".special")) {
                    icon = fileSpecial;
                } else {
                    icon = fileOther;
                }
            }

            setIcon(icon);
        }

        return this;
    }
}

class FileSystemTreeModel implements TreeModel, Comparator<File> {
    private final File                      root;
    private final Vector<TreeModelListener> listeners   = new Vector<>();

    public FileSystemTreeModel(File root) {
        if (!root.isDirectory()) throw new IllegalArgumentException();
        this.root = root;
    }

    @Override
    public int compare(File a, File b) {
        if (a.isDirectory() && !b.isDirectory()) return -1;
        if (!a.isDirectory() && b.isDirectory()) return 1;
        return a.getName().compareToIgnoreCase(b.getName());
    }

    @Override
    public File getRoot() {
        return root;
    }

    @Override
    public File getChild(Object parent, int index) {
        File dir = (File) parent;
        File[] files = dir.listFiles();
        String[] children = new String[files.length];
        Arrays.sort(files, this::compare);

        for (int i = 0; i < files.length; i++) {
            children[i] = files[i].getName();
        }

        return new TreeFile(dir, children[index]);
    }

    @Override
    public int getChildCount(Object parent) {
        File file = (File) parent;
        if (file.isDirectory()) {
            String[] fileList = file.list();
            if (fileList != null) return fileList.length;
        }
        return 0;
    }

    @Override
    public boolean isLeaf(Object node) {
        File file = (File) node;
        return file.isFile();
    }

    @Override
    public int getIndexOfChild(Object parent, Object child) {
        File dir = (File) parent;
        File file = (File) child;
        File[] children = dir.listFiles();
        Arrays.sort(children, this::compare);

        for (int i = 0; i < children.length; i++) {
            if (file.equals(children[i])) return i;
        }

        return -1;
    }

    @Override
    public void valueForPathChanged(TreePath path, Object value) {}

    @Override
    public void addTreeModelListener(TreeModelListener listener) {
        listeners.add(listener);
    }

    @Override
    public void removeTreeModelListener(TreeModelListener listener) {
        listeners.remove(listener);
    }

    private static class TreeFile extends File {
        public TreeFile(File parent, String path) {
            super(parent, path);
        }

        @Override
        public String toString() {
            return getName();
        }
    }
}

【问题讨论】:

  • 1) 为了尽快获得更好的帮助,请发帖 minimal reproducible exampleShort, Self Contained, Correct Example。 2) File Browser GUI 是否显示类似的工件? "(code) which are available here" 我(和大多数人)不会关注代码链接。制作一个 MCVE,将其发布在问题中。 (投票结束。)
  • @AndrewThompson 我只是认为向代码添加链接而不是使这个问题过长会更容易阅读。而且,因为我不了解这个问题的本质,我认为创建 MCVE 可能会非常困难。但是哦,好吧,我想我会创建一个。不过,我可能需要一段时间才能更新问题,因为我很快就要开会了。
  • 我添加了一个 MCVE 来演示我的问题。抄送@AndrewThompson

标签: java swing jtree


【解决方案1】:

也许这只是我昨天遇到的那个。当我选择一个节点时,它后面的节点由与所选节点相同的前景呈现。我找到了一个关于这个问题的规范:UIManagers 目前不能正常工作:键查找正在返回空值。我们正在恢复默认设置。 也许您应该使用 setForeground() 重置未选择的条目。

【讨论】:

    【解决方案2】:

    我能够通过使自定义 TreeCellRenderer 实现 TreeCellRenderer 而不是扩展 DefaultTreeCellRenderer 来解决此渲染问题。似乎问题与 DefaultTreeCellRenderer 扩展 JLabel 并编辑其自己的值以显示条目有关。通过使用将值映射到JLabels 的HashMap,内存消耗会增加很多,但可以解决问题。

    public class FileSystemTreeCellRenderer implements TreeCellRenderer {
        protected Color foregroundColor = null;
        protected Color backgroundColor = null;
        protected Color selectionForegroundColor = null;
        protected Color selectionBackgroundColor = null;
        protected Map<Object, JLabel> labels = new HashMap<>();
        protected final Icon  folderSpecial;
        protected final Icon  folderOther;
        protected final Icon  fileSpecial;
        protected final Icon  fileOther;
    
        public FileSystemTreeCellRenderer() throws IOException {
            InputStream stream;
            BufferedImage image;
    
            stream = new FileInputStream("folder-special.png");
            image = ImageIO.read(stream);
            folderSpecial = new ImageIcon(image);
            stream.close();
    
            stream = new FileInputStream("folder-other.png");
            image = ImageIO.read(stream);
            folderOther = new ImageIcon(image);
            stream.close();
    
            stream = new FileInputStream("file-special.png");
            image = ImageIO.read(stream);
            fileSpecial = new ImageIcon(image);
            stream.close();
    
            stream = new FileInputStream("file-other.png");
            image = ImageIO.read(stream);
            fileOther = new ImageIcon(image);
            stream.close();
        }
    
        protected JLabel getLabelFor(Object object) {
            JLabel label = labels.get(object);
            if(label == null) {
                label = new JLabel();
                labels.put(object, label);
            }
            return label;
        }
    
        public Color getForegroundColor() {
            if (foregroundColor == null) return UIManager.getColor("Tree.textForeground");
            return foregroundColor;
        }
    
        public Color getBackgroundColor() {
            if (backgroundColor == null) return UIManager.getColor("Tree.textBackground");
            return backgroundColor;
        }
    
        public Color getSelectionForegroundColor() {
            if (selectionForegroundColor == null) return UIManager.getColor("Tree.selectionForeground");
            return selectionForegroundColor;
        }
    
        public Color getSelectionBackgroundColor() {
            if (selectionBackgroundColor == null) return UIManager.getColor("Tree.selectionBackground");
            return selectionBackgroundColor;
        }
    
        @Override
        public Component getTreeCellRendererComponent(JTree tree, Object value, boolean selected, boolean expanded, boolean leaf, int row, boolean hasFocus) {
            JLabel label = getLabelFor(value);
            label.setText(Objects.toString(value, ""));
            label.setOpaque(true);
            label.setBackground(selected ? getSelectionBackgroundColor() : getBackgroundColor());
            label.setForeground(selected ? getSelectionForegroundColor() : getForegroundColor());
            label.setEnabled(tree.isEnabled());
            label.setComponentOrientation(tree.getComponentOrientation());
    
            if (value instanceof File) {
                Icon icon = null;
                File file = (File) value;
    
                if (file.isDirectory()) {
                    if (file.getName().endsWith("-special")) {
                        icon = folderSpecial;
                    } else {
                        icon = folderOther;
                    }
                } else if (file.isFile()) {
                    if (file.getName().endsWith(".special")) {
                        icon = fileSpecial;
                    } else {
                        icon = fileOther;
                    }
                }
    
                label.setIcon(icon);
            }
    
            return label;
        }
    }
    

    如果有人有更好、更少内存消耗的方法来做到这一点,请随时发布另一个答案!

    【讨论】:

      猜你喜欢
      • 2013-10-19
      • 2011-10-31
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-04-01
      • 2010-10-19
      • 2016-06-27
      相关资源
      最近更新 更多