gpt4 book ai didi

C#/WPF 位图在设置一些值后保持白色(使用 LockBits)

转载 作者:太空宇宙 更新时间:2023-11-03 10:40:59 25 4
gpt4 key购买 nike

使用 MS Visual Studio 2013,具有以下代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Runtime.InteropServices;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
using System.Drawing;
using System.Drawing.Imaging;

namespace SplashNS
{

public partial class DocumentWindow : Window
{
public SplashFile File;

[DllImport("gdi32")]
static extern int DeleteObject(IntPtr o);

private BitmapSource _loadBitmap(System.Drawing.Bitmap source)
{
IntPtr ip = source.GetHbitmap();
BitmapSource bs = null;
try
{
bs = System.Windows.Interop.Imaging.CreateBitmapSourceFromHBitmap(ip,
IntPtr.Zero, Int32Rect.Empty,
System.Windows.Media.Imaging.BitmapSizeOptions.FromEmptyOptions());
}
finally
{
DeleteObject(ip);
}

return bs;
}

public DocumentWindow()
{
InitializeComponent();
Bitmap canvas = new Bitmap(663, 356);
BitmapData bmd = canvas.LockBits(new System.Drawing.Rectangle(0, 0, canvas.Width, canvas.Height), System.Drawing.Imaging.ImageLockMode.ReadOnly, canvas.PixelFormat);
int PixelSize = 4;

unsafe
{
for (int y = 0; y < canvas.Height; y++)
{
byte* row = (byte*)bmd.Scan0 + (y * bmd.Stride);
for (int x = 0; x < canvas.Width; x++)
{
row[x * PixelSize] = 255;
}
}
}
canvas.UnlockBits(bmd);
SplashCanvas.Source = this._loadBitmap(canvas);
}
}
}

没用。它生成一个白色位图,而不是一个全蓝色像素的位图。没有错误显示。

我想,这一定是菜鸟的问题,但我是初学者。我想它一定是像素格式的东西,但我也可能错了......

最佳答案

您似乎正在创建一个旧式 System.Drawing.Bitmap 然后将其转换为 WPF,但是 WPF 有一套非常完整的实用程序可用于使用 WritableBitmap 从头开始​​创建位图。 .有关概述,请参阅本文:Foundations: Bitmaps and Pixel Bits .

例如,下面是创建给定大小和颜色的实体位图的代码:

public static class BitmapHelper
{
public unsafe static BitmapSource CreateSolidBitmap(int width, int height, double dpiX, double dpiY, Color color)
{
var bitmap = new WriteableBitmap(width, height, dpiX, dpiY, PixelFormats.Pbgra32, null);
var format = bitmap.Format;
int blueIndex, greenIndex, redIndex, alphaIndex;
int bitsPerPixel, bytesPerPixel;
if (!TryParsePixelFormat(format, out bitsPerPixel, out bytesPerPixel, out blueIndex, out greenIndex, out redIndex, out alphaIndex))
return null;
var byteWidth = bytesPerPixel * width;
bitmap.Lock();
try
{
var backBuffer = (byte*)bitmap.BackBuffer.ToPointer();
for (int iRow = 0; iRow < height; iRow++)
{
var row = backBuffer + (iRow * bitmap.BackBufferStride);
for (byte* pixel = row, endRow = row + byteWidth; pixel < endRow; pixel += bytesPerPixel)
{
pixel[blueIndex] = color.B;
pixel[greenIndex] = color.G;
pixel[redIndex] = color.R;
if (alphaIndex >= 0)
pixel[alphaIndex] = color.A;
}
}
bitmap.AddDirtyRect(new Int32Rect(0, 0, width, height));
}
finally
{
bitmap.Unlock();
}
return bitmap;
}

static int BlueIndex = 0;
static int GreenIndex = 1;
static int RedIndex = 2;
static int AlphaIndex = 3;

private static bool TryFindColorIndex(PixelFormatChannelMask mask, out int index)
{
var maskList = mask.Mask;
for (int i = 0, count = maskList.Count; i < count; i++)
{
if (maskList[i] == 255)
{
index = i;
return true;
}
}
index = -1;
return false;
}

static bool TryParsePixelFormat(PixelFormat format, out int bitsPerPixel, out int bytesPerPixel, out int blueIndex, out int greenIndex, out int redIndex, out int alphaIndex)
{
// Currently only implemented for non-indexed formats with 3 or 4 bytes
// per color.
bitsPerPixel = format.BitsPerPixel;
if ((bitsPerPixel % 8) != 0)
{
bytesPerPixel = blueIndex = greenIndex = redIndex = alphaIndex = -1;
return false;
}

bytesPerPixel = bitsPerPixel / 8;

var masks = format.Masks;
int maskCount = masks.Count;

if (maskCount == 3 || maskCount == 4)
{
var blueMask = masks[BlueIndex];
var greenMask = masks[GreenIndex];
var redMask = masks[RedIndex];

if (!TryFindColorIndex(blueMask, out blueIndex)
|| !TryFindColorIndex(greenMask, out greenIndex)
|| !TryFindColorIndex(redMask, out redIndex))
{
bytesPerPixel = blueIndex = greenIndex = redIndex = alphaIndex = -1;
return false;
}

if (maskCount == 3)
{
alphaIndex = -1;
}
else
{
if (!TryFindColorIndex(masks[AlphaIndex], out alphaIndex))
alphaIndex = -1;
}

return true;
}

bytesPerPixel = blueIndex = greenIndex = redIndex = alphaIndex = -1;
return false;
}
}

由于这是一个静态方法,它不知道您正在运行的应用程序的 DPI,但是您可以像这样在主窗口(或任何其他窗口)中获取它:

    PresentationSource source = PresentationSource.FromVisual(this);

double dpiX = 96.0, dpiY = 96.0;
if (source != null)
{
dpiX = 96.0 * source.CompositionTarget.TransformToDevice.M11;
dpiY = 96.0 * source.CompositionTarget.TransformToDevice.M22;
}

96 似乎在不可用时按惯例使用。

关于C#/WPF 位图在设置一些值后保持白色(使用 LockBits),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25347935/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com