【发布时间】:2017-11-07 16:03:10
【问题描述】:
我正在从原始像素数据创建位图,然后使用 ImageSource.FromStream 加载图像。它适用于android和uwp。我使用以下类手动创建位图。
public class BitmapBuilder
{
private const int BitmapTotalHeaderSize = 54;
private const int BitmapHeaderSize = 40;
private const int BitCount = 32;
private const int Planes = 1;
private const int DPI = 96;
private const int ColorsPerChannel = 4;
private static byte[] s_BitmapHeaderByte;
static BitmapBuilder()
{
CreateBitmapHeaderByte();
}
private static void CreateBitmapHeaderByte()
{
s_BitmapHeaderByte = new byte[BitmapTotalHeaderSize];
s_BitmapHeaderByte[0] = (byte)'B';
s_BitmapHeaderByte[1] = (byte)'M';
//Buffer.BlockCopy(BitConverter.GetBytes(buffer.Length), 0, buffer, 2, 4); // File Size
Buffer.BlockCopy(BitConverter.GetBytes(BitmapTotalHeaderSize), 0, s_BitmapHeaderByte, 6, 4); // Data Offset
Buffer.BlockCopy(BitConverter.GetBytes((int)0), 0, s_BitmapHeaderByte, 10, 4); // Padding
Buffer.BlockCopy(BitConverter.GetBytes(BitmapHeaderSize), 0, s_BitmapHeaderByte, 14, 4); // Header Size
//Buffer.BlockCopy(BitConverter.GetBytes((int)width), 0, s_BitmapHeaderByte, 18, 4); // Width
//Buffer.BlockCopy(BitConverter.GetBytes((int)height), 0, s_BitmapHeaderByte, 22, 4); // Height
Buffer.BlockCopy(BitConverter.GetBytes(Planes), 0, s_BitmapHeaderByte, 26, 2); // Planes
Buffer.BlockCopy(BitConverter.GetBytes(BitCount), 0, s_BitmapHeaderByte, 28, 2); // Bit count
Buffer.BlockCopy(BitConverter.GetBytes((int)0), 0, s_BitmapHeaderByte, 30, 4); // Compression
Buffer.BlockCopy(BitConverter.GetBytes((int)0), 0, s_BitmapHeaderByte, 34, 4); // Image Size
Buffer.BlockCopy(BitConverter.GetBytes(DPI), 0, s_BitmapHeaderByte, 38, 4); // XpixelPerM
Buffer.BlockCopy(BitConverter.GetBytes(DPI), 0, s_BitmapHeaderByte, 42, 4); // YpixelPerM
Buffer.BlockCopy(BitConverter.GetBytes((int)0), 0, s_BitmapHeaderByte, 46, 4); // Colors Used
Buffer.BlockCopy(BitConverter.GetBytes((int)0), 0, s_BitmapHeaderByte, 50, 4); // Colors Important
}
public static byte[] GetBitmapFromRawData(IntPtr pixelData, int width, int height)
{
byte[] buffer = new byte[height * width * ColorsPerChannel + BitmapTotalHeaderSize];
Buffer.BlockCopy(s_BitmapHeaderByte, 0, buffer, 0, s_BitmapHeaderByte.Length);
Buffer.BlockCopy(BitConverter.GetBytes(buffer.Length), 0, buffer, 2, 4); // File Size
Buffer.BlockCopy(BitConverter.GetBytes((int)width), 0, buffer, 18, 4); // Width
Buffer.BlockCopy(BitConverter.GetBytes((int)height), 0, buffer, 22, 4); // Height
Marshal.Copy(pixelData, buffer, BitmapTotalHeaderSize, buffer.Length - BitmapTotalHeaderSize); // Write raw data
return buffer;
}
然后我将流设置为 Image 控件
byte[] buffer = BitmapBuilder.GetBitmapFromRawData(pixelData, (int)width, (int)height);
img.Source = ImageSource.FromStream(() => new MemoryStream(buffer));
IOS 不支持。
然后我保存了手动创建的位图并将其用作文件。它适用于 android 和 UWP,但不适用于 IOS。
string sFile = Path.Combine(documentDirectory, "testImage.bmp");
if (File.Exists(sFile))
{
img.Source = ImageSource.FromStream(() => File.OpenRead(sFile));
}
`
我不知道为什么 ImageSource.FromStream 似乎无法在 IOS 上运行。
【问题讨论】:
-
为什么您的位图有 PDF 扩展名?您确定您正在创建的图像是 iOS 支持的格式吗?
-
对不起。复制粘贴错误。已在 testImage.bmp 的代码示例中修复。
-
你确定iOS支持这种图片格式吗?
-
您可以直接创建基于 PNG/JPEG 的 CGImage 并将其用作 UIImage 的背景图像,或者使用 CoreGraphics 从其他支持的格式创建图像,然后使用生成的 CGImage作为背景图片:(developer.apple.com/library/content/documentation/2DDrawing/…
-
我建议创建一个示例应用程序,将您的 bmp 文件直接加载到 UIImage 中,只是为了验证图像格式不是问题
标签: c# xamarin xamarin.ios xamarin.forms imagesource