使用元数据编写图像时,JpegBitmapEncoder.Save()会抛出异常

时间:2014-06-11 06:49:04

标签: c# wpf metadata exif iptc

我正在构建一个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);

有什么想法吗?

2 个答案:

答案 0 :(得分:3)

您的代码中存在一些问题。

  1. 在将BitmapFrame写入目标流之前,源流必须保持打开状态。

  2. 来自BitmapDecoder的BitmapFrame的图像元数据是只读的。当您想要修改元数据时,您必须从原始BitmapFrame创建一个新的BitmapFrame。

  3. 第三个查询似乎被打破了。例外情况说“无法找到财产”。

  4. 此代码适用于我:

    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实例一样,您必须在图像的生命周期内保持流打开。