I have a png file(8indexed and transparent)in the resource,and want to change its fore color and show it in an image.I have tried this :
Image1.Source = new BitmapImage(new Uri($"pack://application:,,,/100.png", UriKind.Absolute));
it works fine without changing the color. Having Searched a lot, I combine some code all from the Internet and get the code below which looks fine,but returns strange reults.
private void Window_Loaded(object sender, RoutedEventArgs e)
{
// read image from base64
const string base64 = "";
string imagebase64 = base64.Substring(base64.IndexOf(",") + 1);
byte[] streamBase = Convert.FromBase64String(imagebase64);
BitmapImage bitmapImage = new BitmapImage();
bitmapImage.BeginInit();
bitmapImage.StreamSource = new MemoryStream(streamBase);
bitmapImage.EndInit();
// change it to Bitmap
Bitmap bmp = ImageSourceToBitmap(bitmapImage);
// change color
for (int x = 0; x < bmp.Width; x++)
{
for (int y = 0; y < bmp.Height; y++)
{
if (bmp.GetPixel(x, y) == Color.Black)
{
bmp.SetPixel(x, y, Color.Blue);
}
}
}
// change it back to BitmapSource
BitmapSource bitmapSource = BitmapToBitmapImage(bmp);
Image1.Source = bitmapSource;
}
// methods below
public static System.Drawing.Bitmap ImageSourceToBitmap(ImageSource imageSource)
{
BitmapSource m = (BitmapSource)imageSource;
System.Drawing.Bitmap bmp = new System.Drawing.Bitmap(m.PixelWidth, m.PixelHeight, System.Drawing.Imaging.PixelFormat.Format8bppIndexed); // 坑点:选Format32bppRgb将不带透明度
System.Drawing.Imaging.BitmapData data = bmp.LockBits(
new System.Drawing.Rectangle(System.Drawing.Point.Empty, bmp.Size), System.Drawing.Imaging.ImageLockMode.WriteOnly, System.Drawing.Imaging.PixelFormat.Format8bppIndexed);
m.CopyPixels(Int32Rect.Empty, data.Scan0, data.Height * data.Stride, data.Stride);
bmp.UnlockBits(data);
return bmp;
}
public BitmapImage BitmapToBitmapImage(Bitmap src)
{
MemoryStream ms = new MemoryStream();
((System.Drawing.Bitmap)src).Save(ms, System.Drawing.Imaging.ImageFormat.Bmp);
BitmapImage image = new BitmapImage();
image.BeginInit();
ms.Seek(0, SeekOrigin.Begin);
image.StreamSource = ms;
image.EndInit();
return image;
}
I wonder whick part makes trouble in it. Sorry for my poor English. Any suggestion would be appreciated!
Update:
- I'm using some weather icons, and the weather website only give me some png files. Path/drawinggroup/streamgeometry sounds great, but it's nearly impossible for me to draw them again in WPF.
- I want to show the image directly, but Stack Overflow tells me "You need at least 10 reputation to post images." emmm....
- Sorry to use fuzzy image, but I want to convey the massage is that not only the exact black should change to the exact blue, the lighter black should also change to lighter blue.
- I make an smaller reproducible example.