如何在SWT中为文件显示系统图标?

11

我想展示一个文件树,类似于java2s.com的'创建懒惰的文件树',但是要包括实际的系统图标,特别是文件夹的图标。 SWT似乎没有提供这个功能(程序API不支持文件夹),所以我想出了下面的解决方案:

public Image getImage(File file)
{
    ImageIcon systemIcon = (ImageIcon) FileSystemView.getFileSystemView().getSystemIcon(file);
    java.awt.Image image = systemIcon.getImage();

    int width = image.getWidth(null);
    int height = image.getHeight(null);
    BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
    Graphics2D g2d = bufferedImage.createGraphics();
    g2d.drawImage(image, 0, 0, null);
    g2d.dispose();
    int[] data = ((DataBufferInt) bufferedImage.getData().getDataBuffer()).getData();
    ImageData imageData = new ImageData(width, height, 24, new PaletteData(0xFF0000, 0x00FF00, 0x0000FF));
    imageData.setPixels(0, 0, data.length, data, 0);
    Image swtImage = new Image(this.display, imageData);
    return swtImage;
}

然而,应该是透明的区域显示为黑色。我该如何解决这个问题,或者是否应该采取其他方法?

更新:

我认为原因在于PaletteData根本不适用于透明度。

现在,我使用Color.WHITE填充BufferedImage,这是一种可接受的解决方法。但我仍然想知道真正的解决方案...

4个回答

7
您需要像下面这样的方法,它是从 http://dev.eclipse.org/viewcvs/index.cgi/org.eclipse.swt.snippets/src/org/eclipse/swt/snippets/Snippet156.java?view=co 复制而来的99%代码:

static ImageData convertToSWT(BufferedImage bufferedImage) {
    if (bufferedImage.getColorModel() instanceof DirectColorModel) {
        DirectColorModel colorModel = (DirectColorModel)bufferedImage.getColorModel();
        PaletteData palette = new PaletteData(colorModel.getRedMask(), colorModel.getGreenMask(), colorModel.getBlueMask());
        ImageData data = new ImageData(bufferedImage.getWidth(), bufferedImage.getHeight(), colorModel.getPixelSize(), palette);
        for (int y = 0; y < data.height; y++) {
            for (int x = 0; x < data.width; x++) {
                int rgb = bufferedImage.getRGB(x, y);
                int pixel = palette.getPixel(new RGB((rgb >> 16) & 0xFF, (rgb >> 8) & 0xFF, rgb & 0xFF)); 
                data.setPixel(x, y, pixel);
                if (colorModel.hasAlpha()) {
                    data.setAlpha(x, y, (rgb >> 24) & 0xFF);
                }
            }
        }
        return data;        
    } else if (bufferedImage.getColorModel() instanceof IndexColorModel) {
        IndexColorModel colorModel = (IndexColorModel)bufferedImage.getColorModel();
        int size = colorModel.getMapSize();
        byte[] reds = new byte[size];
        byte[] greens = new byte[size];
        byte[] blues = new byte[size];
        colorModel.getReds(reds);
        colorModel.getGreens(greens);
        colorModel.getBlues(blues);
        RGB[] rgbs = new RGB[size];
        for (int i = 0; i < rgbs.length; i++) {
            rgbs[i] = new RGB(reds[i] & 0xFF, greens[i] & 0xFF, blues[i] & 0xFF);
        }
        PaletteData palette = new PaletteData(rgbs);
        ImageData data = new ImageData(bufferedImage.getWidth(), bufferedImage.getHeight(), colorModel.getPixelSize(), palette);
        data.transparentPixel = colorModel.getTransparentPixel();
        WritableRaster raster = bufferedImage.getRaster();
        int[] pixelArray = new int[1];
        for (int y = 0; y < data.height; y++) {
            for (int x = 0; x < data.width; x++) {
                raster.getPixel(x, y, pixelArray);
                data.setPixel(x, y, pixelArray[0]);
            }
        }
        return data;
    }
    return null;
}

然后你可以像这样调用它:
static Image getImage(File file) {
    ImageIcon systemIcon = (ImageIcon) FileSystemView.getFileSystemView().getSystemIcon(file);
    java.awt.Image image = systemIcon.getImage();
    if (image instanceof BufferedImage) {
        return new Image(display, convertToSWT((BufferedImage)image));
    }
    int width = image.getWidth(null);
    int height = image.getHeight(null);
    BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);
    Graphics2D g2d = bufferedImage.createGraphics();
    g2d.drawImage(image, 0, 0, null);
    g2d.dispose();
    return new Image(display, convertToSWT(bufferedImage));
}

1
我选择了那种方法。然而,我觉得很遗憾,你必须跳过这么多的障碍才能让它正常工作。 - Jens Bannmann

1

对于文件,您可以使用org.eclipse.swt.program.Program来获取给定文件结尾的图标(具有正确的透明度设置):

File file=...
String fileEnding = file.getName().substring(file.getName().lastIndexOf('.'));
ImageData iconData=Program.findProgram(fileEnding ).getImageData();
Image icon= new Image(Display.getCurrent(), iconData);

对于文件夹,您可以考虑使用静态图标。


0

我还没有详细查看代码,但我注意到你正在使用TYPE_INT_RGB而不是TYPE_INT_ARGB(它包括alpha/透明度支持)。


看到Eclipse网站上的Snippet32,我可以看到通常可以使用 Program 类来获取图标。但是使用扩展名“ .Folder ”似乎不能返回一个实例,尽管它是 getExtensions() 的成员。

这段代码可以获取文件夹图标:

Display display = new Display();
Shell shell = new Shell(display);
Label label = new Label(shell, SWT.NONE);
label.setText("Can't find icon");
Image image = null;
for (Program p : Program.getPrograms()) {
  if ("Folder".equals(p.getName())) {
    ImageData data = p.getImageData();
    if (data != null) {
      image = new Image(display, data);
      label.setImage(image);
    }
    break;
  }
}
label.pack();
shell.pack();
shell.open();
while (!shell.isDisposed()) {
  if (!display.readAndDispatch())
    display.sleep();
}
if (image != null)
  image.dispose();
display.dispose();

那段代码需要改进,我认为,但应该是朝着正确方向的指针。我只在英语版Windows XP上进行了测试。


切换到 TYPE_INT_ARGB 并没有改变任何东西。//我不想使用启发式方法来查找图标。那么各种驱动器呢,比如硬盘、存储棒、DVD 驱动器等等? - Jens Bannmann

0

分享一下代码,对我来说它运行良好。

    /** extract the exe file's icon */
public static Image getImage4exe(String path) {
    /* Use the character encoding for the default locale */
    TCHAR lpszFile = new TCHAR(0, path, true);
    long[] phiconSmall = new long[1];
    OS.ExtractIconEx(lpszFile, 0, null, phiconSmall, 1);
    if (phiconSmall[0] == 0) {
        return null;
    }
    Image ret = Image.win32_new(null, SWT.ICON, phiconSmall[0]);
    return ret;
}

参数值是可执行文件的绝对路径。 - Allen Jiang

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接