我正在创建特定尺寸的占位图像,将用作数据URI
Bitmap bitmap = new Bitmap(16, 10);
我已经做了一些研究,但找不到将此位图保存为尽可能小的文件大小的好方法,这就是我想要一个8位PNG的原因。
我的问题是:如何将此位图保存为文件/字节数组/流成为一个8位PNG?有哪些好的库可以使用?
我正在创建特定尺寸的占位图像,将用作数据URI
Bitmap bitmap = new Bitmap(16, 10);
我已经做了一些研究,但找不到将此位图保存为尽可能小的文件大小的好方法,这就是我想要一个8位PNG的原因。
我的问题是:如何将此位图保存为文件/字节数组/流成为一个8位PNG?有哪些好的库可以使用?
public static bool TryNQuantify(string inputFilename, string outputFilename)
{
var quantizer = new nQuant.WuQuantizer();
var bitmap = new Bitmap(inputFilename);
if (bitmap.PixelFormat != System.Drawing.Imaging.PixelFormat.Format32bppArgb)
{
ConvertTo32bppAndDisposeOriginal(ref bitmap);
}
try
{
using (var quantized = quantizer.QuantizeImage(bitmap))
{
quantized.Save(outputFilename, System.Drawing.Imaging.ImageFormat.Png);
}
}
catch
{
return false;
}
finally
{
bitmap.Dispose();
}
return true;
}
private static void ConvertTo32bppAndDisposeOriginal(ref Bitmap img)
{
var bmp = new Bitmap(img.Width, img.Height, System.Drawing.Imaging.PixelFormat.Format32bppArgb);
using (var gr = Graphics.FromImage(bmp))
gr.DrawImage(img, new Rectangle(0, 0, img.Width, img.Height));
img.Dispose();
img = bmp;
}
更多信息请参见:
我喜欢FreeImage项目,它很轻便且易于使用。以下是创建透明png的示例。您可以轻松地将其包装在一个方法中,并设置宽度、高度和透明度值。
//create a new bit map
FIBITMAP dib = new FIBITMAP();
//allocate a 16x10 bitmap with 8bit depth
dib = FreeImage.Allocate(16, 10, 8);
//set the transpareny
byte[] Transparency = new byte[1];
Transparency[0] = 0x00;
FreeImage.SetTransparencyTable(dib, Transparency);
//save the bitmap
FreeImage.Save(FREE_IMAGE_FORMAT.FIF_PNG, dib, "C:\\temp\\tp.png", FREE_IMAGE_SAVE_FLAGS.DEFAULT);
我通过谷歌搜索找到了一个6年前的帖子,但是我找不到我的解决方案。这个问题在6年前应该已经有解决方法了,因为它使用了.Net Framework的System.Drawing库。
以下示例代码可以帮助任何想要编写8位索引PNG图像的人。
static class SaveImages
{
public static void SaveGrayImage(Bitmap srcImg, string name)
{
Bitmap grayImg = new Bitmap(srcImg.Width, srcImg.Height, PixelFormat.Format8bppIndexed);
var pal = grayImg.Palette;
foreach (int i in Enumerable.Range(0, 256))
pal.Entries[i] = Color.FromArgb(255, i, i, i);
grayImg.Palette = pal;
var data = grayImg.LockBits(new Rectangle(0, 0, srcImg.Width, srcImg.Height), ImageLockMode.WriteOnly, PixelFormat.Format8bppIndexed);
var bytes = new byte[data.Stride * data.Height];
foreach (int y in Enumerable.Range(0, srcImg.Height))
{
foreach (int x in Enumerable.Range(0, srcImg.Width))
{
var colour = srcImg.GetPixel(x, y);
var c = (int)colour.R + (int)colour.G + (int)colour.B;
c /= 3;
bytes[data.Stride * y + x] = (byte)c;
}
}
Marshal.Copy(bytes, 0, data.Scan0, bytes.Length);
grayImg.Save(name, ImageFormat.Png);
}
}
编辑:要增加透明度,请尝试使用调色板。
如果你只需要一个小的透明图像,为什么要止步于8位呢?你可以直接降到1位!反正你只需要一种颜色,而且它会更小。
实际上,你甚至不需要做任何特殊处理。由于新索引位图上的像素都默认为0,这意味着它们引用了第一个调色板颜色,所以你只需要创建一个新的1bpp图像,并将第一个调色板颜色设置为透明即可:
public static Bitmap MakePlaceholderImage(Int32 width, Int32 height)
{
Bitmap bm = new Bitmap(width, height, PixelFormat.Format1bppIndexed);
// This colour can't be assigned directly since the .Palette getter actually makes a copy of the palette.
ColorPalette pal = bm.Palette;
pal.Entries[0] = Color.Transparent;
bm.Palette = pal;
return bm;
}
我对此进行了一些实验,保存为png格式后,最终结果似乎始终比使用8bpp执行的相同代码结果小8倍。对于一个5000x5000像素的图像,以png格式保存的文件大小仅略大于3 KiB。
1) 使用每像素8位的格式创建位图:
Bitmap bmp = new Bitmap(20, 20, System.Drawing.Imaging.PixelFormat.Format8bppIndexed);
请参阅像素格式以获取更多格式信息。
2) 绘制到位图...
3) 然后将其保存为PNG:
bmp.Save(@"c:\temp\xyz.png", System.Drawing.Imaging.ImageFormat.Png);
创建的文件将具有与 bmp
(8位)相同的像素格式。
LockBits
访问/提取后备数组以进行操作。 - Nyerguds