понедельник, 7 декабря 2009 г.

Как создать прозрачное GIF изображение используя C#

Не существует аппаратно легкого пути создания прозрачного GIF изображения используя .NET framework. Microsoft предоставила метод класс Bitmap именуемый MakeTransparent(), но он не работет для GIF-ов, похоже он работет только PNG файлов.

Решение:

Нужно создать цветовую палитру используя спецификацию GIF specification. Для выполнения этого нужно сохранить непрозрачный GIF в MemoryStream, а затем перебрав бинарные данные, обновить цветову палитру новыми "прозрачными" битами.




/// <summary>
/// Returns a transparent background GIF image from the specified Bitmap.
/// </summary>
/// <param name="bitmap">The Bitmap to make transparent.</param>
/// <param name="color">The Color to make transparent.</param>
/// <returns>New Bitmap containing a transparent background gif.</returns>
public Bitmap MakeTransparentGif(Bitmap bitmap, Color color)
{
byte R = color.R;
byte G = color.G;
byte B = color.B;</p>
MemoryStream fin = new MemoryStream();
bitmap.Save(fin, System.Drawing.Imaging.ImageFormat.Gif);
MemoryStream fout = new MemoryStream((int)fin.Length);
int count = 0;
byte[] buf = new byte[256];
byte transparentIdx = 0;
fin.Seek(0, SeekOrigin.Begin);
//header
count = fin.Read(buf, 0, 13);
if ((buf[0] != 71) || (buf[1] != 73) || (buf[2] != 70)) return null; //GIF
fout.Write(buf, 0, 13);
int i = 0;
if ((buf[10] & 0×80) > 0)
{
i = 1 << ((buf[10] & 7) + 1) == 256 ? 256 : 0;
}
for (; i != 0; i—)
{
fin.Read(buf, 0, 3);
if ((buf[0] == R) && (buf[1] == G) && (buf[2] == B))
{
transparentIdx = (byte)(256 – i);
}
fout.Write(buf, 0, 3);
}
bool gcePresent = false;
while (true)
{
fin.Read(buf, 0, 1);
fout.Write(buf, 0, 1);
if (buf[0] != 0×21) break;
fin.Read(buf, 0, 1);
fout.Write(buf, 0, 1);
gcePresent = (buf[0] == 0xf9);
while (true)
{
fin.Read(buf, 0, 1);
fout.Write(buf, 0, 1);
if (buf[0] == 0) break;
count = buf[0];
if (fin.Read(buf, 0, count) != count) return null;
if (gcePresent)
{
if (count == 4)
{
buf[0] |= 0×01;
buf[3] = transparentIdx;
}
}
fout.Write(buf, 0, count);
}
}
while (count > 0)
{
count = fin.Read(buf, 0, 1);
fout.Write(buf, 0, 1);
}
fin.Close();
fout.Flush();
return new Bitmap(fout);
}

1 комментарий:

SpaDe комментирует...

Приветствую, собственно такой вопрос вывел на ваш блог: "создание прозрачного фона в VS08".

Пытаюсь создать приложение с прозрачным фоном. Хочется убрать рамку окна с системным меню - ну это делается легко. Далее, вставляю в форму PNG с прозрачностью. Получаю приложение в котором картика лежит на каком-нибудь однотонном прямоугольнике. - как собственно убрать задний фон формы? то есть сделать его прозрачным на 100%. Key transparency дает очень некрасивый результат, как с прозрачным Gif - ребристые края в областях полупрозрачностей.

Может подскажете?
Спасибо.