将图像的透明区域转换为Alpha通道原理
在Sprite Sheep Editor中,使用了这样的一个小技术(思路来自这里):将透明图像的Alpha通道转换成黑白(灰度)图像,然后与正常图像拼合成一张大图,再存储成JPEG格式。
这方法其实是一个折衷方案。因为JPEG格式是不支持透明的,很多时候为了获得透明效果,我们只能使用PNG格式。但PNG是无损压缩的,在图像尺寸上没有优势。如果将Alpha通道转换成黑白图并保存到JPEG图像中,就能大幅降低最终的图像文件大小。
在我的测试中,一个2000x3300的32位带Alpha通道的PNG图像文件大小为2MB,转换为4000x3300(因为拼合了黑白图片,宽度增加一倍)70%压缩比的JPEG后,文件大小为1.1MB。
当然,转换后的JPEG文件画质比PNG要稍差一些。这可以通过调整压缩比得到一定程度的改善。
要将图像的Alpha通道转换为黑白图像,在AS3中很容易:
1var __p:Point = new Point(0,0);
2_channelBmd = new BitmapData(_bmd.width, _bmd.height, true, 0x00000000);
3_channelBmd.fillRect(_bmd.rect, 0xff000000);
4_channelBmd.copyChannel(_bmd, _bmd.rect, __p, 8, 1);
5_channelBmd.copyChannel(_bmd, _bmd.rect, __p, 8, 2);
6_channelBmd.copyChannel(_bmd, _bmd.rect, __p, 8, 4)
这些代码将 _bmd
的Alpha通道信息分别复制到 _channelBmd
的红绿蓝通道中,最终合成了一张代表Alpha通道的灰度图。
但是,并不是所有语言都有“通道”这个概念。其实AS3中的“通道”(包括Photoshop中的)就是在颜色中的红绿蓝值而已,我们可以手工把ARGB颜色中的Alpha分离出来,将它们组成RGB颜色。这是AS3的实现:
1_pixelBmd = new BitmapData(_bmd.width, _bmd.height, false);
2for(var i:int=0;i<_bmd.height;i++)
3{
4 var __str:String = '';
5 for(var j:int=0;j<_bmd.width;j++)
6 {
7 var __pixel:int = _bmd.getPixel32(j,i);
8 //分离ARGB颜色中的alpha值,alpha处于32位的最前面8位
9 var __alpha:int = $argb >> 24 & 0xFF;
10 var __argbstr:String = __alpha == 0 ? '00' : __alpha.toString(16);
11 __str += __argbstr + ' ';
12 //利用alpha的值合并成RGB颜色值,因为不需要透明,因此24位即可
13 __pixel = __alpha << 16 | __alpha << 8 | __alpha;
14 _pixelBmd.setPixel(j,i, __pixel);
15 }
16 //如果你的屏幕足够宽,可以看到一个字符画,内容你猜猜?
17 trace(__str);
18}
这是JAVA的实现,当然,也可以用RGBImageFilter来做这件事:
1public BufferedImage convertImage(BufferedImage $image)
2{
3 _img = $image;
4 _w = _img.getWidth();
5 _h = _img.getHeight();
6 _sourceARGB = _img.getRGB(0, 0, _w, _h, null, 0, _w);
7
8 BufferedImage __image = new BufferedImage(_w, _h, BufferedImage.TYPE_INT_RGB);
9 int[] __destARGB = new int[_sourceARGB.length];
10 System.out.println("所有数组的长度:"+_sourceARGB.length);
11 for(int i=0;i<_sourceARGB.length;i++)
12 {
13 int __alpha = _sourceARGB[i] >> 24 & 0xFF;
14 __destARGB[i] = __alpha << 16 | __alpha << 8 | __alpha;
15 }
16 __image.setRGB(0, 0, _w, _h, __destARGB, 0, _w);
17 return __image;
18}
一个AS3实现的范例:
- 文章ID:1454
- 原文作者:zrong
- 原文链接:https://blog.zengrong.net/post/convert-transparent-color-to-alpha-channel/
- 版权声明:本作品采用 署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0) 进行许可,非商业转载请注明出处(原文作者,原文链接),商业转载请联系作者获得授权。