- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在开发一个 silverlight 应用程序,其中我所有的图标都是 PNG。
所有这些图标的颜色都是黑色,或者更确切地说是从黑色到灰色,具体取决于 alpha 值。每个 PNG 都有透明背景。
在我的应用程序中,我想逐个像素地从黑色或灰色更改颜色,比方说更改为红色(黑色之前)或浅红色(灰色之前)。
遍历每个像素的字节,我认识到,每个全黑像素的 alpha 值为 255。那些灰色像素的 alpha 值介于 1 和 254 之间。0 的 alpha 值似乎是透明的。黑色的 RGB 值全为 0。
我的想法是,我更改每个像素的颜色,但保留 alpha 值,以便 PNG 保持其原始外观。
为此,我编写了一个小函数,如下所示:
private static WriteableBitmap ColorChange(WriteableBitmap wbmi, System.Windows.Media.Color color)
{
for (int pixel = 0; pixel < wbmi.Pixels.Length; pixel++)
{
if (wbmi.Pixels[pixel] != 0)
{
byte[] colorArray = BitConverter.GetBytes(wbmi.Pixels[pixel]);
byte blue = colorArray[0];
byte green = colorArray[1];
byte red = colorArray[2];
byte alpha = colorArray[3];
if (alpha > 0)
{
//HERE, I change the color, but keep the alpha
//
wbmi.Pixels[pixel] = Gui.Colors.ToInt.Get(color,alpha);
colorArray = BitConverter.GetBytes(wbmi.Pixels[pixel]);
blue = colorArray[0];
green = colorArray[1];
red = colorArray[2];
alpha = colorArray[3];
String colorString = "blue: " + blue.ToString() + "green: " + green.ToString() + "red: " + red.ToString() + "alpha: " + alpha.ToString();
System.Diagnostics.Debug.WriteLine(colorString);
}
}
}
return wbmi;
}
还有一个小助手类来改变颜色但保持 alpha:
namespace Gui.Colors
{
public class ToInt
{
public static int Get(System.Windows.Media.Color color, Byte alpha)
{
color.A = alpha;
return color.A << 24 | color.R << 16 | color.G << 8 | color.B;
}
}
}
我的问题是,我可以清楚地看到,alpha 小于 255(灰色)的像素是问题所在。尽管保留了 alpha 值并且颜色发生了变化,但像素看起来不一样了,它们不再透明了。这使得边缘看起来很粗糙。
长话短说:看看我的代码——问题是什么?如何使用原始 PNG 的透明度实现逐像素颜色转换?
请原谅我的英语。我的母语不是英语。
最佳答案
这个案子解决了。我找到了 Rene Schulte 写的 WriteableBitmapExtensions 类。
此扩展类提供此功能(以及更多...):
private const float PreMultiplyFactor = 1 / 255f;
public static void SetPixeli(this WriteableBitmap bmp, int index, byte a, Color color)
{
float ai = a * PreMultiplyFactor;
bmp.Pixels[index] = (a << 24) | ((byte)(color.R * ai) << 16) | ((byte)(color.G * ai) << 8) | (byte)(color.B * ai);
}
改变颜色的例程现在看起来像这样:
/// <summary>
/// Changes the color of every pixel in a WriteableBitmap with an alpha value > 0 to the desired color.
/// </summary>
/// <param name="wbmi"></param>
/// <param name="color"></param>
/// <returns></returns>
private static WriteableBitmap ColorChange(WriteableBitmap wbmi, System.Windows.Media.Color color)
{
for (int pixel = 0; pixel < wbmi.Pixels.Length; pixel++)
{
byte[] colorArray = BitConverter.GetBytes(wbmi.Pixels[pixel]);
byte blue = colorArray[0];
byte green = colorArray[1];
byte red = colorArray[2];
byte alpha = colorArray[3];
if (alpha > 0)
{
wbmi.SetPixeli(pixel, alpha, color);
}
}
wbmi.Invalidate();
return wbmi;
}
您可以在此处下载 Rene Schulte 的作品:
http://dotnet.dzone.com/sites/all/files/WriteableBitmapExtensions.zip
整个项目可以在这里找到:
http://writeablebitmapex.codeplex.com/
由于没有人在 SO 上回答这个问题,大炮向 Rene Schulte 致敬。 YMMD! :)
顺便说一下:我正在使用 NounProject 的 SVG ,将它们转换为所需大小的 PNG,然后我使用此方法为它们着色。由于 NounProject 中的所有符号都是黑色的,因此这是获得高质量符号的好方法。
关于c# - 逐像素颜色转换 WriteableBitmap => PNG Black only to Color with Transparency,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16382536/
为什么这个脚本不起作用?仅当页面宽度超过 915 像素时,我希望单击按钮时滚动页面 100 像素。我试图通过仅在宽度超过 915 像素时允许该函数运行来实现此目的,但它没有发生。怎么办? $(docu
我需要您帮助我建立一个网站。我是一个新手,已经得到了一个设计为 900 像素宽的主体,但他们给了我一个 1200 像素宽的图像。他们希望图像跨越整个 1200 像素的宽度,因此页面两侧基本上会有 30
我有一个在 y 轴上展开的 UIScrollview 并调用这个委托(delegate)方法: -(void)scrollViewDidScroll:(UIScrollView *)scrollVie
我有一个固定的标题这个标题在我滚动时改变高度和图像标志但是当我调整窗口大小时我希望图像保持比例但随着我缩小浏览器而变得更小标志只有在限制时缩小浏览器靠近图像,但我希望在调整浏览器大小时图像变小。 我该
在我的项目中,我使用 ArcGIS API for JavaScript https://developers.arcgis.com/javascript/但是对于(在这里插入非常大的坏词)我无法覆盖
有没有办法使用 jQuery,根据窗口滚动的距离做不同的事情? 这是我现在使用的代码; $(document).scroll(function() { // If scroll distanc
这基本上是 Jetpack Joyride 中运动的基本版本,但不是 Joyrider 以每秒 100 像素的速度下降,而是字母“x”从控制台的正中间以每秒 100 像素的速度下降和点击事件会导致它以
我像这样处理 MINMAXINFO: case WM_GETMINMAXINFO: { LPMINMAXINFO p_info = (LPMINMAXINFO)lPar
我对 javascript 有点陌生,我一直在查找 documentElement、clientWidth 和 clientHeight 并试图找出为什么它将我的 Canvas 设置为 300px x
我正在编写一些软件来读取 DICOM 文件,但我不确定如何处理具有未定义长度的标签。标准是这样说的 “如果值字段具有显式长度,则值长度字段应包含等于长度(以字节为单位)的值 值字段。否则,值字段 有一
我对 OpenGL 有点陌生,但我很确定我的问题在于所使用的像素格式,或者我的纹理是如何生成的...... 我正在使用 16 位 RGB5_A1 像素格式在平面 2D 四边形上绘制纹理,但在这个阶段我
有没有办法获取直播电视流,例如在像素级别上进行分析。 我的目标是检查直播电视流(例如使用java),例如广播电台 Logo 是否可见。 有机会通过 Google 电视观看此直播吗? 是否有机会通过笔记
我正在尝试构建一个函数,它以给定角度从特定坐标延伸,并循环遍历该线上的像素,直到遇到黑色像素. 如果角度为 180 度,这很容易实现。在这种情况下,搜索只会向下扩展,在每次迭代中将列坐标加 1。然而,
我已经研究了一段时间,但找不到任何解决方案。 这是我的代码 如果您将此代码复制并粘贴到本网站的 HTML 区域:http://jsfiddle.net/T3Nnu/3/ 如果您查看 Facebo
我有一个网页 - http://bit.ly/YHFX5B如果你看一下页脚,你会发现它后面有一些额外的白色像素/线条。我不明白他们是从哪里来的。 请告知他们可能来自哪里。 谢谢,丹 最佳答案 在 #f
如何在没有状态栏和操作栏的情况下获取屏幕高度(像素)或者如果有人告诉我如何获取状态栏和操作栏的高度,它也会有所帮助。我已经找到了屏幕高度,但它包括状态栏和操作栏.我将支持库 v7 用于操作栏。我在网上
Java 字符串根据宽度(像素)换行 在一些场景下,我们经常会通过判断字符串的长度,比如个数来实现换行,但是中文、英文、数字、其实在展示的时候同样长度的字符串,其实它的宽度是不一样的,这也是们我通
我创建了一个不错的简单可扩展列表。它应该像单选列表一样工作,您应该只能选择一个元素。我还没有实现这部分,因为我对列表的大小有疑问: class ExpandableListRadio extends
我使用以下代码滚动到元素顶部,但我想滚动到元素顶部上方 10px,不知道如何执行此操作,有什么建议吗?谢谢! $('html, body').stop(true,true).animate({
我有一个链接,可以在滚动时更改其垂直位置。当我点击此链接时,我想(平滑地)转到页面上的某个位置,该位置距离页面顶部正好 1080 像素。 我无法实现它,希望有人能帮助我。 链接: 脚本: $(do
我是一名优秀的程序员,十分优秀!