我有一个只包含彩色形状和透明背景的GIF图像
我想用我想要的颜色替换形状的颜色(这个gif的颜色调色板只有2种颜色:在我的情况下是透明的和白色的。)
我创建了一个过滤器,用红色正确替换白色(这是一个测试)
但是我的方法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);
}
}
答案 0 :(得分:2)
您的代码中存在3个问题:
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)您正在使用BufferedImage bufferedImage
(默认)清除imageToBufferedImage(...)
中Color.BLACK
的颜色。将//g2.setBackground(Color.blue);
替换为g2.setBackground(new Color(0, 0, 0, 0));
或删除g2.clearRect(...);
答案 1 :(得分:0)
确定您的图片支持Alpha通道? 尝试首先将非黑色全尺寸矩形绘制到bufferedImage?
答案 2 :(得分:0)
猜猜:
a)也许您不需要clearRect(...)
方法。
b)也许你可以使用类似的东西:
g2.setColor(new Color(0, 0, 0, 0));
g2.fillRect(...);
答案 3 :(得分:0)
你有这个:
g2.drawImage(image, 0, 0, null);
从这个方法的Javadoc:
图像中的透明像素不会影响已存在的任何像素。
由于您清除了图像并用背景颜色填充,因此像素已经是黑色,因此它们会保持黑色。
尝试
g2.setBackground(new Color(0, 0, 0, 0) );
g2.clearRect(0, 0, 200, 40);
在绘制图像之前。
答案 4 :(得分:0)
使用复合清除像素。您不能只在其他像素或任何垃圾上绘制透明像素。 Porter-Duff规则很明确,任何这样做都会导致像素根据复合规则混合。所以只需更改,规则就会清除像素。
问题是大多数解决方案都试图使用规则将图像哄骗到最终获得你想要的像素。更改规则一秒钟,一切都变得更容易。
Composite composite = g.getComposite();
g.setComposite(AlphaComposite.getInstance(AlphaComposite.CLEAR));
g.fillRect(0, 0, bufferedimage.getWidth(), bufferedimage.getHeight());
g.setComposite(composite);
这将清除当前像素的图形对象。然后再画回你想要的东西。我的例子恢复了之前的复合,因为你会惊讶地发现这些东西经常变得奇怪,这样你就可以获得清晰的像素并且可以重新开始而不用图形上下文改变任何东西。