您可以在位图图像中将一种颜色更改为另一种颜色吗?

本文关键字:颜色 一种 另一种 位图 图像 | 更新日期: 2023-09-27 18:11:48

对于Bitmap,有MakeTransparent的方法,是否有类似的将一种颜色更改为另一种颜色?

// This sets Color.White to transparent
Bitmap myBitmap = new Bitmap(sr.Stream);
myBitmap.MakeTransparent(System.Drawing.Color.White);

有什么东西可以做到这一点吗?

Bitmap myBitmap = new Bitmap(sr.Stream);
myBitmap.ChangeColor(System.Drawing.Color.Black, System.Drawing.Color.Gray);

您可以在位图图像中将一种颜色更改为另一种颜色吗?

通过对Yorye Nathan的评论的好奇,这是我通过修改http://msdn.microsoft.com/en-GB/library/ms229672(v=vs.90).aspx创建的扩展。

它可以将位图中的所有像素从一种颜色变为另一种颜色。

public static class BitmapExt
{
    public static void ChangeColour(this Bitmap bmp, byte inColourR, byte inColourG, byte inColourB, byte outColourR, byte outColourG, byte outColourB)
    {
        // Specify a pixel format.
        PixelFormat pxf = PixelFormat.Format24bppRgb;
        // Lock the bitmap's bits.
        Rectangle rect = new Rectangle(0, 0, bmp.Width, bmp.Height);
        BitmapData bmpData =
        bmp.LockBits(rect, ImageLockMode.ReadWrite,
                     pxf);
        // Get the address of the first line.
        IntPtr ptr = bmpData.Scan0;
        // Declare an array to hold the bytes of the bitmap. 
        // int numBytes = bmp.Width * bmp.Height * 3; 
        int numBytes = bmpData.Stride * bmp.Height;
        byte[] rgbValues = new byte[numBytes];
        // Copy the RGB values into the array.
        Marshal.Copy(ptr, rgbValues, 0, numBytes);
        // Manipulate the bitmap
        for (int counter = 0; counter < rgbValues.Length; counter += 3)
        {
            if (rgbValues[counter] == inColourR &&
                rgbValues[counter + 1] == inColourG &&
                rgbValues[counter + 2] == inColourB)
             {
                rgbValues[counter] = outColourR;
                rgbValues[counter + 1] = outColourG;
                rgbValues[counter + 2] = outColourB;
             }
        }
        // Copy the RGB values back to the bitmap
        Marshal.Copy(rgbValues, 0, ptr, numBytes);
        // Unlock the bits.
        bmp.UnlockBits(bmpData);
    }
}

bmp.ChangeColour(0,128,0,0,0,0);调用

从这个答案中提取代码:

public static class BitmapExtensions
{
    public static Bitmap ChangeColor(this Bitmap image, Color fromColor, Color toColor)
    {
        ImageAttributes attributes = new ImageAttributes();
        attributes.SetRemapTable(new ColorMap[]
        {
            new ColorMap()
            {
                OldColor = fromColor,
                NewColor = toColor,
            }
        }, ColorAdjustType.Bitmap);
        using (Graphics g = Graphics.FromImage(image))
        {
            g.DrawImage(
                image,
                new Rectangle(Point.Empty, image.Size),
                0, 0, image.Width, image.Height,
                GraphicsUnit.Pixel,
                attributes);
        }
        return image;
    }
}

虽然我没有对它进行基准测试,但这应该比在循环中执行GetPixel/SetPixel的任何解决方案都要快。它也更直接一些。

你可以使用SetPixel:

private void ChangeColor(Bitmap s, System.Drawing.Color source, System.Drawing.Color target)
{
    for (int x = 0; x < s.Width; x++)
    {
        for (int y = 0; y < s.Height; y++)
        {
            if (s.GetPixel(x, y) == source)
                s.SetPixel(x, y, target);
        }                
    }
}

GetPixel和SetPixel是gdiplus.dll函数GdipBitmapGetPixel和GdipBitmapSetPixel的包装器

备注:

根据位图的格式,GdipBitmapGetPixel可能不会返回与GdipBitmapSetPixel设置相同的值。例如,如果你在一个Bitmap对象上调用GdipBitmapSetPixel,它的像素格式是32bppPARGB,像素的RGB组件是预相乘的。一个随后对GdipBitmapGetPixel的调用可能会返回一个不同的值因为四舍五入。同样,如果你在位图上调用GdipBitmapSetPixel对象的颜色深度为每像素16位,则信息可以是在从32位到16位的转换和随后的调用期间丢失GdipBitmapGetPixel可能会返回一个不同的值。

您需要一个库来提供一种无需处理像素就可以修改图像颜色空间的方法。LeadTools有一个非常广泛的图库,你可以使用它来支持颜色空间修改,包括交换颜色。