我正在构建一个WPF桌面应用,以帮助我整理照片以发布到Facebook。这是我在新位置创建照片副本的代码,其中添加了标题(EXIF + IPTC + XMP):
private void SaveImageAs(string currPath, string newPath, bool setCaption = false, string captionToSet = "")
{
System.IO.FileStream stream = new System.IO.FileStream(currPath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite);
JpegBitmapDecoder decoder = new JpegBitmapDecoder(stream, BitmapCreateOptions.None, BitmapCacheOption.None);
BitmapFrame bitmapFrame = decoder.Frames[0];
BitmapMetadata metadata = bitmapFrame.Metadata.Clone() as BitmapMetadata;
stream.Close();
if (setCaption)
{
// if we want to set the caption, do it in EXIF, IPTC, and XMP
metadata.SetQuery("/app1/ifd/{uint=270}", captionToSet);
metadata.SetQuery("/app13/irb/8bimiptc/iptc/Caption", captionToSet);
metadata.SetQuery("/xmp/dc:description/x-default", captionToSet);
}
MemoryStream memstream = new MemoryStream(); // create temp storage in memory
JpegBitmapEncoder encoder = new JpegBitmapEncoder();
encoder.Frames.Add(BitmapFrame.Create(bitmapFrame, bitmapFrame.Thumbnail, metadata, bitmapFrame.ColorContexts));
encoder.Save(memstream); // save in memory
stream.Close();
stream = new FileStream(newPath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.ReadWrite);
memstream.Seek(0, System.IO.SeekOrigin.Begin); // go to stream start
byte[] bytes = new byte[memstream.Length + 1];
memstream.Read(bytes, 0, (int)memstream.Length);
stream.Write(bytes, 0, bytes.Length);
stream.Close();
memstream.Close();
}
运行它,我得到一个" COMException未处理"异常突出显示这一行:
encoder.Save(memstream);
未处理的类型' System.Runtime.InteropServices.COMException'发生在 PresentationCore.dll中
其他信息:句柄无效。 (HRESULT异常:0x80070006(E_HANDLE))
我看到here这可能是由于线程问题造成的,所以我没有直接从应用中调用SaveImageAs,而是添加了这个,但没有效果:
private void _SaveImageAs(string currPath, string newPath, bool setCaption = false, string captionToSet = "")
{
Thread saveThread = new Thread(() => SaveImageAs(currPath, newPath, setCaption, captionToSet));
saveThread.SetApartmentState(ApartmentState.STA);
saveThread.IsBackground = false;
saveThread.Start();
}
我还尝试更换MemoryStream来创建一个本地临时文件的FileStream--它没有改变任何东西:
FileStream memstream = new FileStream(System.IO.Path.GetDirectoryName(newPath) + @"\" + "temp.jpg", System.IO.FileMode.OpenOrCreate);
有什么想法吗?
答案 0 :(得分:3)
您的代码中存在一些问题。
在将BitmapFrame写入目标流之前,源流必须保持打开状态。
来自BitmapDecoder的BitmapFrame的图像元数据是只读的。当您想要修改元数据时,您必须从原始BitmapFrame创建一个新的BitmapFrame。
第三个查询似乎被打破了。例外情况说“无法找到财产”。
此代码适用于我:
public static void SaveImageAs(string sourcePath, string targetPath, string caption)
{
using (var sourceStream = new FileStream(sourcePath, FileMode.Open, FileAccess.Read, FileShare.Read))
{
var decoder = new JpegBitmapDecoder(sourceStream, BitmapCreateOptions.None, BitmapCacheOption.None);
var frame = decoder.Frames[0];
if (!string.IsNullOrWhiteSpace(caption))
{
frame = BitmapFrame.Create(frame);
var metadata = (BitmapMetadata)frame.Metadata;
metadata.SetQuery("/app1/ifd/{uint=270}", caption);
metadata.SetQuery("/app13/irb/8bimiptc/iptc/Caption", caption);
//metadata.SetQuery("/xmp/dc:description/x-default", caption);
}
var encoder = new JpegBitmapEncoder();
encoder.Frames.Add(frame);
using (var targetStream = new FileStream(targetPath, FileMode.Create))
{
encoder.Save(targetStream);
}
}
}
答案 1 :(得分:1)
你得到了例外,因为你关闭了用于加载原始jpeg的流。注释第一个stream.Close()(正好在if(setCaption)之上)并且它将起作用。就像使用Image实例一样,您必须在图像的生命周期内保持流打开。