Как рисовать прозрачное изображение с помощью System.Drawing? - PullRequest
11 голосов
/ 10 октября 2008

Я пытаюсь вернуть прозрачный GIF со страницы .aspx для отображения на веб-странице. Я пытаюсь, чтобы изображение имело прозрачность, но я просто продолжаю получать черное, где изображение должно быть прозрачным.

Кто-нибудь знает, что я делаю не так?

Protected Sub Page_Load(ByVal sender As Object, ByVal e As System.EventArgs) _
  Handles Me.Load
    '' Change the response headers to output a GIF image.
    Response.Clear()
    Response.ContentType = "image/gif"

    Dim width = 110
    Dim height = width

    '' Create a new 32-bit bitmap image
    Dim b = New Bitmap(width, height)

    '' Create Grahpics object for drawing
    Dim g = Graphics.FromImage(b)

    Dim rect = New Rectangle(0, 0, width - 1, height - 1)

    '' Fill in with Transparent
    Dim tbrush = New System.Drawing.SolidBrush(Color.Transparent)
    g.FillRectangle(tbrush, rect)

    '' Draw Circle Border
    Dim bPen = Pens.Red
    g.DrawPie(bPen, rect, 0, 365)

    '' Fill in Circle
    Dim cbrush = New SolidBrush(Color.LightBlue)
    g.FillPie(cbrush, rect, 0, 365)


    '' Clean up
    g.Flush()
    g.Dispose()

    '' Make Transparent
    b.MakeTransparent()

    b.Save(Response.OutputStream, Imaging.ImageFormat.Gif)
    Response.Flush()
    Response.End()
End Sub

Ответы [ 4 ]

7 голосов
/ 10 октября 2008

Да, как сказал Джером, в любом случае создавать прозрачные GIF-объекты с помощью растрового объекта не существует. Дерьмо!

Ну, во всяком случае, я изменил свой код для генерации PNG, и все работает как положено.

Есть одна небольшая работа, которую мне нужно было сделать, поскольку вы не можете записывать PNG напрямую в OutputStream. Мне нужно было записать PNG в MemoryStream, а затем записать это в OutputStream.

Вот окончательный код моей реализации:

Protected Sub Page_Load(ByVal sender As Object, ByVal e As System.EventArgs) _
  Handles Me.Load
    '' Change the response headers to output a JPEG image.
    Response.Clear()
    Response.ContentType = "image/png"

    Dim width = 11
    Dim height = width

    '' Create a new 32-bit bitmap image
    Dim b = New Bitmap(width, height)

    '' Create Grahpics object for drawing
    Dim g = Graphics.FromImage(b)

    '' Fill the image with a color to be made Transparent after drawing is finished.
    g.Clear(Color.Gray)

    '' Get rectangle where the Circle will be drawn
    Dim rect = New Rectangle(0, 0, width - 1, height - 1)

    '' Draw Circle Border
    Dim bPen = Pens.Black
    g.DrawPie(bPen, rect, 0, 365)

    '' Fill in Circle
    Dim cbrush = New SolidBrush(Color.Red)
    g.FillPie(cbrush, rect, 0, 365)

    '' Clean up
    g.Flush()
    g.Dispose()

    '' Make Transparent
    b.MakeTransparent(Color.Gray)

    '' Write PNG to Memory Stream then write to OutputStream
    Dim ms = New MemoryStream()
    b.Save(ms, Imaging.ImageFormat.Png)
    ms.WriteTo(Response.OutputStream)

    Response.Flush()
    Response.End()
End Sub
5 голосов
/ 10 октября 2008

К сожалению, не существует простого способа создания прозрачного Gif-объекта с использованием растрового объекта. (См. эту статью KB )

Вы также можете использовать формат PNG, который поддерживает прозрачность с кодом, который вы используете.

4 голосов
/ 10 октября 2008

Это возможно , но не просто
Если вы можете использовать небезопасный код в своем проекте, есть несколько способов использовать указатели для копирования таблицы цветов и обеспечения прозрачности.

Образец приложения для форм Боба Пауэлла доступен здесь http://www.bobpowell.net/giftransparency.htm. Несколько лет назад я использовал вариант этого метода в веб-обработчике, который получал доступ около 10 миллионов раз в месяц, и, похоже, он работал хорошо.

Если вы используете только ограниченную цветовую палитру, вы можете уменьшить обработку таблицы цветов до нужных вам цветов (точно не помню, как я это сделал ...).

Как уже говорилось, png - это метрическая чушь проще.

2 голосов
/ 26 января 2010

вот некоторый код для преобразования gif (в котором уже есть прозрачность) (предполагается, что вы хотите изменить его размер) в растровом изображении, а затем его можно правильно отобразить с помощью прозрачности.

imagePath = System.Web.HttpContext.Current.Request.MapPath(libraryPath + reqImageFile);
System.Drawing.Image image = null;
Bitmap resizedImage = null;

if (reqWidth == 0) { reqWidth = image.Width; }
if (reqHeight == 0) { reqHeight = image.Height; }
image = System.Drawing.Image.FromFile(imagePath);
reqWidth = image.Width;
reqHeight = image.Height;

//here is the transparency 'special' treatment
resizedImage = new Bitmap(reqWidth, reqHeight, PixelFormat.Format8bppIndexed);
ColorPalette pal = resizedImage.Palette;
for (int i = 0; i < pal.Entries.Length; i++)
{
       Color col = pal.Entries[i];
       pal.Entries[i] = Color.FromArgb(0, col.R, col.G, col.B);
}
resizedImage.Palette = pal;
BitmapData src = ((Bitmap)image).LockBits(new Rectangle(0, 0, reqWidth, reqHeight),  ImageLockMode.ReadOnly, image.PixelFormat);
BitmapData dst = resizedImage.LockBits(new Rectangle(0, 0, resizedImage.Width, resizedImage.Height),
ImageLockMode.WriteOnly, resizedImage.PixelFormat);
((Bitmap)image).UnlockBits(src);
resizedImage.UnlockBits(dst);

Удачи!

Grégoire Lafortune

...