Java背景透明

5

我有一张gif图像,其中只包含一个有颜色的形状和一个透明的背景。

我想要用我想要的颜色替换形状的颜色(这个gif的调色板只有两种颜色:透明和白色,在我的情况下)。

我创建了一个过滤器,可以正确地将白色替换为红色(这是一个测试)。

但是,我在使用imageToBufferedImage方法时遇到了一个问题,它会删除透明度并将其替换为黑色(不知道为什么)。

所以,我目前所做的就是:

import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.Image;
import java.awt.Toolkit;
import java.awt.image.BufferedImage;
import java.awt.image.FilteredImageSource;
import java.awt.image.ImageFilter;
import java.awt.image.ImageProducer;
import java.awt.image.RGBImageFilter;
import java.io.File;
import javax.imageio.ImageIO;

public class TestPNG {
    
    public static void main(String[] args) throws Exception {

        File in = new File("bg.gif");
        BufferedImage source = ImageIO.read(in);
        int color = source.getRGB(0, 0);

        Image image = makeColorTransparent(source, new Color(color), new Color(255, 0, 0));

        BufferedImage transparent = imageToBufferedImage(image);

        File out = new File("bg2.gif");
        ImageIO.write(transparent, "gif", out);

    }

    private static BufferedImage imageToBufferedImage(Image image) {
        BufferedImage bufferedImage = new BufferedImage(image.getWidth(null), image.getHeight(null), BufferedImage.TYPE_INT_ARGB);
        Graphics2D g2 = bufferedImage.createGraphics();
        //g2.setBackground(Color.blue);
        g2.clearRect(0, 0, 200, 40);
        g2.drawImage(image, 0, 0, null);
        g2.dispose();
        return bufferedImage;
    }

    public static Image makeColorTransparent(BufferedImage im, final Color search, final Color replace) {
        ImageFilter filter = new RGBImageFilter() {
                public final int filterRGB(int x, int y, int rgb) {
                        if (rgb == search.getRGB()) {
                            return replace.getRGB();
                        } else {
                            return rgb;
                        }
                }
        };
        ImageProducer ip = new FilteredImageSource(im.getSource(), filter);
        return Toolkit.getDefaultToolkit().createImage(ip);
    }

}

我需要指定,即使删除过滤器仍会导致黑色而不是透明度的相同问题,因此该问题可能来自Toolkit.getDefaultToolkit().createImage(im.getSource())或imageToBufferedImage方法。 - Bouki
5个回答

2

你的代码存在三个问题:

1)替换

Image image = makeColorTransparent(source, new Color(color), new Color(255, 0, 0));

使用

Image image = makeColorTransparent(source, color, new Color(255, 0, 0));

并且

public static Image makeColorTransparent(BufferedImage im, final Color search, final Color replace) {
...
if (rgb == search.getRGB()) {
...
}

使用

public static Image makeColorTransparent(BufferedImage im, final int search, final Color replace) {
...
if (rgb == search) {
...
}

由于某些原因,source.getRGB(0, 0)忽略了alpha值并变成了白色((255, 255, 255, 0)变为了(255, 255, 255, 255))。

2)不能使用int color = source.getRGB(0, 0),因为它使用第一个像素的颜色(透明)。你应该使用其他代码(比如在控制台询问)来查找要存储在int color中的像素颜色。

3)你正在imageToBufferedImage(...)方法中用Color.BLACK(默认值)清除BufferedImage bufferedImage的颜色。//g2.setBackground(Color.blue);替换为g2.setBackground(new Color(0, 0, 0, 0));或删除g2.clearRect(...);


0

只是猜测:

a)也许你不需要使用clearRect(...)方法。

b)也许你可以使用类似以下的内容:

g2.setColor(new Color(0, 0, 0, 0));
g2.fillRect(...);

0

你有这个:

    g2.drawImage(image, 0, 0, null);

从此方法的Javadoc中:

图像中的透明像素不会影响已经存在的任何像素。

由于您清除了图像并用背景颜色填充它,因此像素已经是黑色,所以它们保持为黑色。

尝试一下

g2.setBackground(new Color(0, 0, 0, 0) );
g2.clearRect(0, 0, 200, 40);

在绘制图像之前。


做这个不起作用,它会从颜色板中移除透明度,所以我的图像现在完全是红色的。 - Bouki
@Bouki:你能提供这个gif吗? - Mark Peters
测试所用的图片可以在这里找到: 链接 - Bouki

0

你可以通过使用Composite清除像素。不能仅仅在其他像素上绘制透明像素或任何垃圾。Porter-Duff规则很清楚,这样做会根据Composite规则使像素混合。因此,只需更改规则并清除像素即可。

问题是大多数解决方案都试图使用规则说服图像最终获得所需的像素。稍微改变一下规则,一切都会变得容易得多。

Composite composite = g.getComposite();

g.setComposite(AlphaComposite.getInstance(AlphaComposite.CLEAR));
g.fillRect(0, 0, bufferedimage.getWidth(), bufferedimage.getHeight());

g.setComposite(composite);

这将清除当前像素的图形对象。然后只需重新绘制您想要的内容。我的示例恢复了先前的组合,因为您会惊讶地发现这些东西经常出现问题,这样您就可以获得清除的像素,并可以在不更改图形上下文中的任何内容的情况下重新开始。


0

确定你的图像支持 alpha 通道吗? 先尝试在 bufferedImage 中绘制一个非黑色的全尺寸矩形。


测试所用的图片可以在这里找到: 链接 - Bouki
我的意思是这一行代码:Image image = makeColorTransparent(source, new Color(color), new Color(255, 0, 0)); 在这里你只使用了RGB颜色,所以我认为不支持alpha通道。 - Thomas
添加alpha参数对问题没有任何影响。 - Bouki

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