如何从剪贴板class获取图像格式?

How to get the image format from Clipboard class?

我正在用 C# 制作图像查看器。该功能是在 C# 上使用剪贴板 class 的复制和粘贴功能。

并且我成功获取了BitmapSource的图像。但它无法检查图像格式(jpeg、png、bmp)。

这是受支持的 C# 源代码 BitmapSource source = Clipboard.GetImage();

这些是我想要使用的。 byte[] image = Clipboard.GetImage("image/png") byte[] image = Clipboard.GetImage("image/bmp")

我会说,如何检查来自 BitmapSource 或剪贴板的图像格式?

简单地说,你不能。

图像作为原始位图保存在剪贴板上。一些应用程序也把它放在png格式,但不是为了保留原件,只是to better support alpha-transparency

但是,一般来说,剪贴板根本不包含该图像的来源或原始格式的任何指示。您还必须意识到剪贴板上的图像可能根本就不是文件。这就像假设任何复制的文本片段必须是一个完整的文本文件,具有文件名。该图像可能是直接从某些编辑器复制的,甚至可能是直接的 [PrintScreen] 按钮屏幕截图。

然而...

注意到当我从 Chrome 复制图像并将其粘贴到 Discord 中时,Discord 以某种方式知道它在 Chrome 中的文件名.这是因为您可以同时将多种格式的数据放入剪贴板。

这个系统通常是为了确保每个应用程序都能读取最适合它的内容。例如,对于从网页复制的内容,记事本将首选纯文本版本,而 Microsoft Word 将使用 HTML(如果可用)。同样,正如我提到的,应用程序经常复制位图和 png 格式的图像,以确保接收支持透明度的应用程序可以采用 transparency-supporting PNG 版本,而不知道使用 PNG 的新趋势的应用程序可以采用经典位图。

但是很多应用程序也使用它来将元数据添加到复制的内容中。当我在 Chrome 中复制你头像的 32x32 缩略图时,剪贴板中的其中一件事是这个被识别为 HTML Format:

的文本片段
Version:0.9
StartHTML:0000000105
EndHTML:0000000238
StartFragment:0000000141
EndFragment:0000000202
<html>
<body>
<!--StartFragment--><img src="https://i.stack.imgur.com/Eql2x.jpg?s=32&amp;g=1"/><!--EndFragment-->
</body>
</html>

(我相信指定 HTML 片段信息的 header 可能是标准格式。不过,我从未研究过。)

如您所见,这包含文件名,从中您通常会得到一个扩展名,可用于确定文件的原始格式。

但是文件不再是那种格式。因为,它只是剪贴板上的数据,而不是实际文件。复制图像后 Chrome 剪贴板的完整转储包含以下格式:

  • System.Drawing.Bitmap(.Net 位图 object。这在 WPF 上的显示可能不同)
  • Bitmap(我觉得和System.Drawing.Bitmap一样)
  • HTML Format(上面显示的片段)
  • DeviceIndependentBitmap(包含 DIB v1 图像字节的字节流;在剪贴板上传输图像的最常见方式;often abused as alpha-capable because it's 32 bit RGB
  • Format17(包含DIB v5图像字节的字节流;比v1稍微先进,并正式支持alpha。'17'指的是旧[=69中使用的旧数字剪贴板格式=] 版本)

如您所见,其中没有 jpeg 文件。因为当您处于剪贴板级别时,永远不会假定原始文件格式会被保留。即使是设法读取并保留文件名的 Discord,也会将粘贴的文件保存为 png,并根据需要调整文件扩展名。

请注意,Chrome 就是这样做的。无法保证来自其他应用程序,甚至来自其他浏览器的剪贴板副本将发送元数据,如果它们发送,是否采用稍微类似的格式。


我用来分析剪贴板数据的代码:

(注意,这是 Windows 表单代码。我不知道它的 WPF 等价物)

DataObject retrievedData = (DataObject)Clipboard.GetDataObject();
if (retrievedData == null)
    return;
String[] formats = retrievedData.GetFormats();
foreach (String format in formats)
{
    Object contents = retrievedData.GetData(format);
    MemoryStream ms = contents as MemoryStream;
    Byte[] bContents = ms == null ? null : ms.ToArray();
    String sContents = contents as String;

    // Check if bContents and sContents are null here, and analyse their contents

    // ...
}