我正在尝试实现我在网上找到的模糊功能,但出于某种原因,它将图像绘制为完全白色。我非常确定原始代码段的功能并不完全正确,因此错误。
以下是最简单的代码形式:
public void Blur(int blurSize)
{
// look at every pixel in the blur rectangle
for (int xx = 0; xx < this.Width; xx++)
{
for (int yy = 0; yy < this.Height; yy++)
{
float avgR = 0;
float avgG = 0;
float avgB = 0;
float avgA = 0;
int blurPixelCount = 0;
// average the color of the red, green and blue for each pixel in the
// blur size while making sure you don't go outside the image bounds
for (int x = xx; (x < xx + blurSize && x < this.Width); x++)
{
for (int y = yy; (y < yy + blurSize && y < this.Height); y++)
{
Color pixel = this.GetPixel(x, y);
avgR += pixel.R;
avgG += pixel.G;
avgB += pixel.B;
avgA += pixel.A;
blurPixelCount++;
}
}
avgR = avgR / blurPixelCount;
avgG = avgG / blurPixelCount;
avgB = avgB / blurPixelCount;
avgA = avgA / blurPixelCount;
// now that we know the average for the blur size, set each pixel to that color
for (int x = xx; x < xx + blurSize && x < this.Width; x++)
{
for (int y = yy; y < yy + blurSize && y < this.Height; y++)
{
SetPixel(x, y, new Color(avgR, avgG, avgB, avgA));
}
}
}
}
}
这是在&#39;尝试&#39;之后模糊(由于Stackoverflow为白色,我将背景更改为黑色):
我注意到如果blurSize大于1,那么图像就像白色一样(使用blurSize 2,背景透明度由我制作黑色)。
可以想象,任何高于2的东西都会产生纯白色图像。如果它不明显,这就是我的目标:
我的问题非常简单,有人能发现这种模糊功能的错误吗?
更新
根据要求,这是我的自定义位图类:
using System;
using System.Collections.Generic;
using System.Drawing;
using System.Drawing.Imaging;
using System.Linq;
using System.Runtime.InteropServices;
using System.Security.Cryptography.X509Certificates;
using System.Text;
using System.Threading.Tasks;
using GrimoireEngine.Framework.Maths;
using Point = GrimoireEngine.Framework.Maths.Point;
using Rectangle = GrimoireEngine.Framework.Maths.Rectangle;
namespace GrimoireEngine.Framework.Utilities
{
public class GrimoireBitmap
{
public Bitmap Source;
private IntPtr _iptr = IntPtr.Zero;
public BitmapData BitmapData;
public byte[] Pixels { get; set; }
public int Depth { get; private set; }
public int Width { get; private set; }
public int Height { get; private set; }
public Color this[int x, int y]
{
get { return GetPixel(x, y); }
set { SetPixel(x, y, value); }
}
public Color this[Point point]
{
get { return GetPixel(point); }
set { SetPixel(point, value); }
}
public bool IsLocked { get; private set; }
public GrimoireBitmap(int width, int height)
{
this.Source = new Bitmap(width, height);
}
public GrimoireBitmap(int width, int height, PixelFormat format)
{
this.Source = new Bitmap(width, height, format);
}
public GrimoireBitmap(Bitmap image)
{
this.Source = image;
}
public GrimoireBitmap(string file)
{
this.Source = new Bitmap(file);
}
/// <summary>
/// Lock bitmap data
/// </summary>
public void LockBits()
{
Width = Source.Width;
Height = Source.Height;
int pixelCount = Width * Height;
System.Drawing.Rectangle rect = new System.Drawing.Rectangle(0, 0, Width, Height);
Depth = Bitmap.GetPixelFormatSize(Source.PixelFormat);
if (Depth != 8 && Depth != 24 && Depth != 32)
{
throw new ArgumentException("Only 8, 24 and 32 bpp images are supported.");
}
BitmapData = Source.LockBits(rect, ImageLockMode.ReadWrite, Source.PixelFormat);
int step = Depth / 8;
Pixels = new byte[pixelCount * step];
_iptr = BitmapData.Scan0;
Marshal.Copy(_iptr, Pixels, 0, Pixels.Length);
this.IsLocked = true;
}
/// <summary>
/// Unlock bitmap data
/// </summary>
public void UnlockBits()
{
Marshal.Copy(Pixels, 0, _iptr, Pixels.Length);
Source.UnlockBits(BitmapData);
this.IsLocked = false;
}
/// <summary>
/// Get the color of the specified pixel
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <returns></returns>
public Color GetPixel(int x, int y)
{
Color clr = Color.Transparent;
int cCount = Depth / 8;
int i = ((y * Width) + x) * cCount;
if (i > Pixels.Length - cCount)
throw new IndexOutOfRangeException();
if (Depth == 32) // For 32 bpp get Red, Green, Blue and Alpha
{
byte b = Pixels[i];
byte g = Pixels[i + 1];
byte r = Pixels[i + 2];
byte a = Pixels[i + 3]; // a
clr = Color.FromNonPremultiplied(a, r, g, b);
}
if (Depth == 24) // For 24 bpp get Red, Green and Blue
{
byte b = Pixels[i];
byte g = Pixels[i + 1];
byte r = Pixels[i + 2];
clr = Color.FromNonPremultiplied(r, g, b);
}
if (Depth == 8) // For 8 bpp get color value (Red, Green and Blue values are the same)
{
byte c = Pixels[i];
clr = Color.FromNonPremultiplied(c, c, c);
}
return clr;
}
/// <summary>
///
/// </summary>
/// <param name="point"></param>
/// <returns></returns>
public Color GetPixel(Point point)
{
return GetPixel(point.X, point.Y);
}
/// <summary>
/// Set the color of the specified pixel
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="color"></param>
public void SetPixel(int x, int y, Color color)
{
int i = (((y * Width) + x) * (Depth / 8));
if (Depth == 32) // For 32 bpp set Red, Green, Blue and Alpha
{
Pixels[i] = color.B;
Pixels[i + 1] = color.G;
Pixels[i + 2] = color.R;
Pixels[i + 3] = color.A;
}
if (Depth == 24) // For 24 bpp set Red, Green and Blue
{
Pixels[i] = color.B;
Pixels[i + 1] = color.G;
Pixels[i + 2] = color.R;
}
if (Depth == 8) // For 8 bpp set color value (Red, Green and Blue values are the same)
{
Pixels[i] = color.B;
}
}
/// <summary>
///
/// </summary>
/// <param name="point"></param>
/// <param name="color"></param>
public void SetPixel(Point point, Color color)
{
SetPixel(point.X, point.Y, color);
}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="width"></param>
/// <param name="height"></param>
/// <param name="color"></param>
public void FillRectangle(int x, int y, int width, int height, Color color)
{
for (int i = 0; i < width; i++)
{
for (int j = 0; j < height; j++)
{
SetPixel(i + x, j + y, color);
}
}
}
/// <summary>
///
/// </summary>
/// <param name="rectangle"></param>
/// <param name="color"></param>
public void FillRectangle(Rectangle rectangle, Color color)
{
FillRectangle(rectangle.X, rectangle.Y, rectangle.Width, rectangle.Height, color);
}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="bitmap"></param>
public void DrawBitmap(int x, int y, GrimoireBitmap bitmap)
{
for (int i = 0; i < bitmap.Width; i++)
{
for (int j = 0; j < bitmap.Height; j++)
{
SetPixel(x + i, y + j, bitmap.GetPixel(i, j));
}
}
}
/// <summary>
///
/// </summary>
/// <param name="color"></param>
public void Fill(Color color)
{
FillRectangle(0, 0, this.Width, this.Height, color);
}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="width"></param>
/// <param name="height"></param>
/// <returns></returns>
public GrimoireBitmap CopyRegion(int x, int y, int width, int height)
{
GrimoireBitmap bitmap = new GrimoireBitmap(width, height);
bitmap.LockBits();
for (int i = 0; i < width; i++)
{
for (int j = 0; j < height; j++)
{
bitmap.SetPixel(i, j, this.GetPixel(x + i, y + j));
}
}
bitmap.UnlockBits();
return bitmap;
}
/// <summary>
///
/// </summary>
/// <param name="rectangle"></param>
/// <returns></returns>
public GrimoireBitmap CopyRegion(Rectangle rectangle)
{
return CopyRegion(rectangle.X, rectangle.Y, rectangle.Width, rectangle.Height);
}
/// <summary>
///
/// </summary>
public void Clear()
{
for (int i = 0; i < this.Width; i++)
{
for (int j = 0; j < this.Height; j++)
{
SetPixel(i, j, Color.Transparent);
}
}
}
/// <summary>
///
/// </summary>
/// <param name="fileName"></param>
public void Save(string fileName)
{
this.Source.Save(fileName);
}
/// <summary>
///
/// </summary>
/// <param name="fileName"></param>
public void Load(string fileName)
{
this.Source = new Bitmap(fileName);
}
/// <summary>
///
/// </summary>
/// <returns></returns>
public Bitmap ToBitmap()
{
return this.Source;
}
/// <summary>
///
/// </summary>
/// <param name="image"></param>
public void FromBitmap(Bitmap image)
{
this.Source = image;
}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="width"></param>
/// <param name="height"></param>
/// <param name="color"></param>
/// <param name="amount"></param>
public void Blend(int x, int y, int width, int height, Color color, float amount)
{
for (int i = 0; i < width; i++)
{
for (int j = 0; j < height; j++)
{
SetPixel(i + x, j + y, Color.Blend(GetPixel(i + x, j + y), color, amount));
}
}
}
/// <summary>
///
/// </summary>
/// <param name="rectangle"></param>
/// <param name="color"></param>
/// <param name="amount"></param>
public void Blend(Rectangle rectangle, Color color, float amount)
{
Blend(rectangle.X, rectangle.Y, rectangle.Width, rectangle.Height, color, amount);
}
public void Blur(int blurSize)
{
// look at every pixel in the blur rectangle
for (int xx = 0; xx < this.Width; xx++)
{
for (int yy = 0; yy < this.Height; yy++)
{
float avgR = 0;
float avgG = 0;
float avgB = 0;
float avgA = 0;
int blurPixelCount = 0;
// average the color of the red, green and blue for each pixel in the
// blur size while making sure you don't go outside the image bounds
for (int x = xx; (x < xx + blurSize && x < this.Width); x++)
{
for (int y = yy; (y < yy + blurSize && y < this.Height); y++)
{
Color pixel = this.GetPixel(x, y);
avgR += pixel.R;
avgG += pixel.G;
avgB += pixel.B;
avgA += pixel.A;
blurPixelCount++;
}
}
avgR = avgR / blurPixelCount;
avgG = avgG / blurPixelCount;
avgB = avgB / blurPixelCount;
avgA = avgA / blurPixelCount;
// now that we know the average for the blur size, set each pixel to that color
for (int x = xx; x < xx + blurSize && x < this.Width; x++)
{
for (int y = yy; y < yy + blurSize && y < this.Height; y++)
{
SetPixel(x, y, new Color(avgR, avgG, avgB, avgA));
}
}
}
}
}
}
}
答案 0 :(得分:1)
编辑: 我修改了代码以使用System.Drawing.Bitmap。它工作正常,这是Blur = 5的结果:
以下是代码:
public void Blur(int blurSize, Bitmap input)
{
// look at every pixel in the blur rectangle
for (int xx = 0; xx < input.Width; xx++)
{
for (int yy = 0; yy < input.Height; yy++)
{
float avgR = 0;
float avgG = 0;
float avgB = 0;
float avgA = 0;
int blurPixelCount = 0;
// average the color of the red, green and blue for each pixel in the
// blur size while making sure you don't go outside the image bounds
for (int x = xx; (x < xx + blurSize && x < input.Width); x++)
{
for (int y = yy; (y < yy + blurSize && y < input.Height); y++)
{
Color pixel = input.GetPixel(x, y);
avgR += pixel.R;
avgG += pixel.G;
avgB += pixel.B;
avgA += pixel.A;
blurPixelCount++;
}
}
avgR = avgR / blurPixelCount;
avgG = avgG / blurPixelCount;
avgB = avgB / blurPixelCount;
avgA = avgA / blurPixelCount;
// now that we know the average for the blur size, set each pixel to that color
for (int x = xx; x < xx + blurSize && x < input.Width; x++)
{
for (int y = yy; y < yy + blurSize && y < input.Height; y++)
{
input.SetPixel(x, y, Color.FromArgb((int)avgA, (int)avgR, (int)avgG, (int)avgB));
}
}
}
}
}
因此,其中一种方法(如SetPixel或GetPixel或新Color)无法正常工作。