- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试创建一个 PNG 文件来保存我们捕获的图像数据,但在尝试设置公司、CameraManufacturer、亮度等元数据时我真的很痛苦。
Google PNGCS
库可以执行此操作,但我必须先编写文件,然后重新加载并重新保存。
我们使用的是 WPF,所以我认为我可以使用更通用的类,如 BitmapMetadata
、PngBitmapEncoder
、JpegBitmapEncoder
。
但是,我不断遇到 为我想使用的属性抛出类型为“System.NotSupportedException”
的异常。您可以在我放在本文底部创建的元数据中的变量 watch
中看到这一点。
Image 类实际上是一个包含宽度高度、文件类型(如 png 或 gif 等)的数据结构。
从摄像头获取图像数据,添加我们想要的标签并保存到文件应该很简单。
与负载相同 - 我们应该能够获得它们。
代码:
/// <summary>
/// Handles the load, save, and export of images
/// </summary>
public interface IImageProvider
{
string GetPath(string fileNameWithoutExtension, ImageVersion version);
string GetPath(Plate plate);
Image Load(string path);
Image Load(string path, int width);
Image Load(Plate plate);
Image LoadThumb(Plate plate);
Task<Image> LoadAsync(Plate plate);
Task<Image> LoadAsync(Plate plate, ImageVersion version);
void Save(Image image, string path, Resolution resolution);
void Save(Image image, string name, ImageCategories category, Resolution resolution);
void Save(Plate plate, Image image, Resolution resolution);
Task SaveAsync(Image image, string path, Resolution resolution);
void Export(Image image, string name, ExportFormats format, string directory, Resolution resolution);
void ExportAsync(Image image, string path, ExportFormats format, string directory, double increment, Resolution resolution);
Task<Image> Import(string fileName, Project project);
void Delete(Plate plate);
Image Load(Plate plate, IStage stage);
}
class ImageProvider
{
private void Save(Image image, string name, ExportFormats format, string directory, Resolution resolution)
{
BitmapEncoder bitmapEncoder = null;
//Determine the type of the export
switch (format)
{
case ExportFormats.bmp:
bitmapEncoder = new BmpBitmapEncoder();
break;
case ExportFormats.jpg:
bitmapEncoder = new JpegBitmapEncoder();
break;
case ExportFormats.png:
bitmapEncoder = new PngBitmapEncoder();
break;
case ExportFormats.tiff:
bitmapEncoder = new TiffBitmapEncoder();
break;
}
if (bitmapEncoder != null)
{
var source = ResizeToResolutionUniform(image, resolution);
ReadOnlyCollection<ColorContext> colorContexts = null;
BitmapMetadata metadata = new BitmapMetadata("png");
BitmapSource thumbnail = null;
BitmapFrame bitmapFrame = BitmapFrame.Create(source, thumbnail, metadata, colorContexts);
bitmapEncoder.Frames.Add(bitmapFrame);
//Get the folder and extension
var extension = Enum.GetName(typeof(ExportFormats), format);
var path = Path.Combine(directory, name + "." + extension);
//Create the directory if needed
if (!string.IsNullOrEmpty(directory) && !Directory.Exists(directory))
Directory.CreateDirectory(directory);
if (image.ProcessingSettings != null)
{
BitmapMetadata bmp = bitmapEncoder.Frames[0].Metadata as BitmapMetadata;
BitmapMetadata bitmapMetadata = (BitmapMetadata)bitmapEncoder.Frames[0].Metadata;//bitmapEncoder.Metadata;
if (!AddMetatDataTags(bitmapMetadata, image.ProcessingSettings))
LogW($"Could not to {path}");
}
using (FileStream fileStream = new FileStream(path, FileMode.Create))
{
//Save the image
bitmapEncoder.Save(fileStream);
}
}
}
public bool AddMetatDataTags(BitmapMetadata bitmapMetadata, IImageProcessingSettings settings)
{
CultureInfo culture = CultureInfo.InvariantCulture;
var ret = false;
if (bitmapMetadata != null)
{
// 1 off invariant information
// Note: the PNG specification does not support many of these metadata properties (which are based on EXIF, which is NOT by the PNG spec)
bitmapMetadata.SetQuery("/tEXt/Author", Environment.UserName); //Environment.UserName;
object obj = bitmapMetadata.GetQuery("/tEXt/Author");
string s = obj.ToString();
AddKey(bitmapMetadata, 1, "CameraManufacturer", "Singer Instrument Company Limited");
AddKey(bitmapMetadata, 2, "CameraModel", "Phenobooth");
AddKey(bitmapMetadata, 0, "ApplicationName", "PhenoSuite");
AddKey(bitmapMetadata, 3, "Brightness", Convert.ToString(settings.Brightness, culture)); // Capture specific data
AddKey(bitmapMetadata, 4, "Exposure", Convert.ToString(settings.Exposure, culture));
AddKey(bitmapMetadata, 5, "Gain", Convert.ToString(settings.Gain, culture));
//...
ret = true;
}
return ret;
}
/*
* The tag dictionary in the bitmap properties has a strange implementation - based on a separated key value pair
* if n = 0 entry like: /iTXt/Keyword = key /iTXt/TextEntry = val
* if n > 0 entry like: /[n]iTXt/Keyword = key /[n]iTXt/TextEntry = val
*/
private void AddKey(BitmapMetadata metaData, int n, string key, string val)
{
var _key = string.Format($"iTXt/{0}{1}{2}", (n > 0) ? "[" : "", (n > 0) ? n.ToString() : "", (n > 0) ? "]" : "");
try
{
metaData.SetQuery(_key + "Keyword", key.ToCharArray()); // need to convert using ToCharArray as internal representation is based on the LPSTR C type
metaData.SetQuery(_key + "TextEntry", val.ToCharArray());
}
catch (Exception e)
{
LogE($"Could not add metadata key:{key} index: {n} {e.Message}");
throw;
}
}
public class Image
{
#region Constructors
/// <summary>
/// Creates an image optionally copying the metadata
/// </summary>
/// <param name="width"></param>
/// <param name="height"></param>
/// <param name="format"></param>
/// <param name="_imageMetadata"></param>
public Image(int width, int height, PixelFormat format, ImageMetadata _imageMetadata) : this()
{
Width = width;
Height = height;
Format = format;
ImageMetadata = _imageMetadata;
}
...
}
当我查看创建的元数据时,有许多不受支持的属性 -事实上,我已经尝试了各种方法来解决这个问题,我确信这曾经有效,但我无法弄清楚发生了什么变化。
新元数据对象上的 watch
示例:Meta Data
最佳答案
这是我用来从 iTXt png 元数据中获取键值对的代码
public static class ImageUtils
{
/// <summary>
/// Captures all or part of the raw png metadata.
/// Can use this to capture PhonoBooth metadata by setting the filter to "iTXt"
///
/// throws: ArgumentException if not a png file
/// </summary>
/// <param name="imageFilePath"></param>
/// <param name="itemMap"></param>
/// <param name="filter"> optional filter on the key (contains)</param>
/// <returns>true if successful, false otherwise.</returns>
public static bool GetMetaDataItems(string imageFilePath, ref Dictionary<string, string> itemMap, string filter=null)
{
Assertion<ArgumentException>(imageFilePath.ToLower().EndsWith(".png"), "Expected png file");
var ret = false;
var query = string.Empty;
itemMap.Clear();
try
{
using (Stream fileStream = File.Open(imageFilePath, FileMode.Open))
{
var decoder = BitmapDecoder.Create(fileStream, BitmapCreateOptions.PreservePixelFormat, BitmapCacheOption.None);
GetMetaDataItems(decoder.Frames[0].Metadata as BitmapMetadata, ref itemMap, filter);
}
ret = true;
}
catch (Exception e)
{
ret = false;
LogE(e.Message);
}
return ret;
}
/// <summary>
/// Used to get the meta data from png file metadata
/// Can use this to capture PhonoBooth metadata by setting the filter to "iTXt"
/// </summary>
/// <param name="bitmapMetadata"></param>
/// <param name="itemMap"></param>
/// <param name="filter">set this to iTXt for Phenosuite image data</param>
/// <param name="query">initally null, used in recursive calls to get the child data</param>
public static void GetMetaDataItems(BitmapMetadata bitmapMetadata , ref Dictionary<string, string> itemMap, string filter= null, string query = null )
{
if (query == null)
query = string.Empty;
if (bitmapMetadata != null)
{
var key = string.Empty;
foreach (string relativeQuery in bitmapMetadata)
{
var fullQuery = query + relativeQuery;
// GetQuery returns an object: either a string or child metadata
// If a string then it is one of 4 values: ["Keyword", "Translated", "Compression", "Language Tag", "TextEntry"]
// We want the Keyword and the subsequent TextEntry items, the tags are a sequence in the order specified above
var metadata = bitmapMetadata.GetQuery(relativeQuery);
var innerBitmapMetadata = metadata as BitmapMetadata;
if (innerBitmapMetadata == null)
AddToMap(ref key, fullQuery, metadata?.ToString(), ref itemMap, filter); // Not a metadata structure so it is data - therefore check and Add to map
else
GetMetaDataItems(innerBitmapMetadata, ref itemMap, filter, fullQuery); // Recursive call
}
}
}
/// <summary>
/// Suitable for Png iTXt metadata
/// This is used to buld the item map from the metadata
/// </summary>
/// <param name="key">key like "Application" or "Lighting Mode"</param>
/// <param name="fullQuery">metadata query</param>
/// <param name="metadata">image metadata</param>
/// <param name="itemMap">map being populated from the metadata</param>
/// <param name="filter">we dont want all the meta data - so this filters on the "sub folder" of the meta data -Phenosuite uses "iTXt" </param>
private static void AddToMap(ref string key, string fullQuery, string metadata, ref Dictionary<string, string> itemMap, string filter)
{
if (metadata != null)
{
if (!fullQuery.Contains("Translated"))
{
if ((filter == null) || ((fullQuery.Contains(filter))))
{
if (fullQuery.Contains("Keyword"))
key = metadata;
if (fullQuery.Contains("TextEntry") && (key != null))
itemMap[key] = metadata?.ToString();
}
}
}
}
}
希望这对某些人有所帮助 - 因为我发现它很难理解!
关于c# - 如何使用元数据创建 png 文件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47163090/
我正在寻找一种经济合理的解决方案来长时间存储图片。我读到 PNG 文件格式,它与 JPEG 相比具有优越的特性,即在这些类别中: 无专利、无许可、无版税 无质量损失 尚未压缩 我有很多来自 Photo
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 9年前关闭。 Improve this q
我怎么能用 FFMEG 做到这一点? 输入 : 背景图片:background.png 图片帧列表:image_001.png,image_002.png ...(每张图片为一帧) 输出:所有帧都有背
$ cat png.ll ./packages/apps/Mms/res/mipmap-hdpi/ic_launcher_smsmms.png ./packages/apps/Mms/res/draw
这个问题在这里已经有了答案: Natural Sort Order in C# (18 个答案) 关闭 7 年前。 这是我的代码: private void Method1() { int
我一直在考虑用 Webp 图像替换我的 Android 应用程序中的 png 文件以减小 APK 大小。 虽然结果不错,但我想知道我是否使用了一些 png 转换器/压缩器,并且能够将尺寸减小到比我为
在 gnuplot-4.2.6 中,我可以使用 set term png medium x000000 xffffff set output 'file.png' plot x 这将生成一个带有黑色背
背景: 我正在努力使一堆 PNG 尽可能小。我正在使用诸如 PngOut、PngCrush 和 OptiPng 之类的工具。 问题: 我遇到了一个大小为 1434 KB 但只有 230 x 230 像
我正在使用 ImageMagick 调整图像大小。如果我传递 -resize WxH 选项,它会按预期运行。但是如果我通过 -resize WxH! (在调整大小时忽略纵横比),一些图像,尤其是 PN
如何访问/删除 PNG 元数据? 我正在寻找 Mac 应用程序或 PHP 代码段。 最佳答案 抱歉发布了一个 Windows 软件,但如果你没有找到任何对 MAC 有用的东西,那就是 TweakPNG
到目前为止似乎没有任何效果。我看到了 pnginfo以下消息: concept_Sjet_dream6.png... Image Width: 200 Image Length: 240 Bi
我有一个带有 Alpha channel (即透明度)的 PNG 图像,我需要创建将图像层合成到白色背景上的版本。我想使用可编写脚本的命令,使用 CLI 工具(例如 Image Magick)将 PN
我是初学者。我昨天问了一个类似的问题,但不知何故被否决了。所以这次我尽量简化问题。 带有 alpha png 的 24 位与 32 位 png 相同吗? 非常感谢您的一些提示。 最佳答案 没有“24
我有这个带点的荷兰 pdf 图像: pdf image of the netherlands with dots 当我尝试将此 pdf 转换为 png 图像时,使用 pdftools和 png像这样:
我在我的启动图像通用项目中添加了“Default.png,Default-568h@2x.png,Default@2x.png”这三个文件,我有三个不同的图像,分辨率与苹果中提到的完全相同文档,适用于
我在 Python 中使用 google app engine 并有几个静态 .png 图像文件,但它们都以“image/x-png”内容类型提供。这是一个问题,当我使用像 chrome 这样的浏览器
我做了一个 python 脚本,该脚本设法根据特定模式解散乱序(png)图像,该 python 脚本使用 ffmpeg 并进行 12 次编码来解乱它(通过裁剪特定部分并将其粘贴到现有图片上)。 因此,
我有一个 PNG 图像文件。我想将其转换为 GeoTiff。我安装了 QGIS 软件,但无法使用它,也不知道如何对图像进行地理配准。请帮我。有没有在线软件? 最佳答案 这是一个非常好的教程,其中包含有
我有一堆使用我编写的 Java 图表工具创建的图表 - 它们主要是黑白图表,带有浅绿色的块,偶尔还有其他颜色。它们当前被保存为 JPG 文件,我想将它们插入到我准备按需打印的书中。 这本书是一个 Op
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
我是一名优秀的程序员,十分优秀!