2015-03-13 17 views
6

Potrzebuję dodać znacznik metadanych (opis) do przesłanych obrazów.Zapisywanie metadanych do jpg i png

Znalazłem tę odpowiedź: https://stackoverflow.com/a/1764913/6776, która działa doskonale dla plików JPG, ale nie dla PNG.

private string Tag = "test meta data"; 

private static Stream TagImage(Stream input, string type) 
{ 
    bool isJpg = type.EndsWith("jpg", StringComparison.InvariantCultureIgnoreCase) || type.EndsWith("jpeg", StringComparison.InvariantCultureIgnoreCase); 
    bool isPng = type.EndsWith("png", StringComparison.InvariantCultureIgnoreCase); 

    BitmapDecoder decoder = null; 

    if (isJpg) 
    { 
     decoder = new JpegBitmapDecoder(input, BitmapCreateOptions.PreservePixelFormat, BitmapCacheOption.OnLoad); 
    } 
    else if (isPng) 
    { 
     decoder = new PngBitmapDecoder(input, BitmapCreateOptions.PreservePixelFormat, BitmapCacheOption.OnLoad); 
    } 
    else 
    { 
     return input; 
    } 

    // modify the metadata 
    BitmapFrame bitmapFrame = decoder.Frames[0]; 
    BitmapMetadata metaData = (BitmapMetadata)bitmapFrame.Metadata.Clone(); 
    metaData.Subject = Tag; 
    metaData.Comment = Tag; 
    metaData.Title = Tag; 

    // get an encoder to create a new jpg file with the new metadata.  
    BitmapEncoder encoder = null; 
    if (isJpg) 
    { 
     encoder = new JpegBitmapEncoder(); 
    } 
    else if (isPng) 
    { 
     encoder = new PngBitmapEncoder(); 
    } 

    encoder.Frames.Add(BitmapFrame.Create(bitmapFrame, bitmapFrame.Thumbnail, metaData, bitmapFrame.ColorContexts)); 

    // Save the new image 
    Stream output = new MemoryStream(); 
    encoder.Save(output); 

    output.Seek(0, SeekOrigin.Begin); 

    return output; 
} 

Działa świetnie, kiedy przesłać jpg, ale z png, na linii metaData.Subject = Tag, to rzuca System.NotSupportedException (ten kodek nie obsługuje określonej właściwości).

Aktualizacja

Wydaje muszę użyć innej metody opartej na Format obrazu:

if (isJpg) 
{ 
    metaData.SetQuery("/app1/ifd/exif:{uint=270}", Tag); 
} 
else 
{ 
    metaData.SetQuery("/tEXt/{str=Description}", Tag); 
} 

podstawie the available formats' queries pierwszy powinien pracować dla obu formatów. Drugi tak naprawdę nie działa (tworzy metadane na obrazie, ale nie zapisuje jego wartości).

Jeśli spróbuję użyć pierwszej metody (/app1/ifd/exif) dla PNG, na linii encoder.Save pojawia się nieobsługiwany wyjątek "brak komponentu obrazowania odpowiedniego".

+0

niezwiązane z pytaniem, ale Myślę, że w twoim poleceniu isJpg = jest błąd. Zakładam, że chcesz przetestować pod kątem ".jpg" lub ".jpeg", ale dwa razy testujesz ".jpg". – RenniePet

+0

Tak, zostało to naprawione w kodzie od tego czasu, ale nie w pytaniu. Dzięki! – thomasb

Odpowiedz

-1

PNG format nie obsługuje metadane :(

Ale XMP robi, co może być pomocne przy konwersji pomiędzy JPEG, EXIF-z metadanych i PNG.

+0

Zgodnie z tematem [Portable Network Graphics] (https://en.wikipedia.org/wiki/Portable_Network_Graphics#Ancillary_chunks) w Wikipedii, PNG może przechowywać metadane. Więc nie jestem pewien, dlaczego uważasz, że nie może. –

+1

Podejrzewam, że jest tak, ponieważ element Metadata PngBitmapEncoder nie jest ustawiony. W szczególności 'encoder.Metadata = new BitmapMetadata (" png ");' generuje wyjątek "Określony BitmapEncoder nie obsługuje globalnych metadanych." – Eponymous

2

I rozwiązać go za pomocą pngcs biblioteki (trzeba zmienić nazwę pobranego dll do „pngcs.dll”)

Oto jak I wdrożone go:

using Hjg.Pngcs; // https://code.google.com/p/pngcs/ 
using Hjg.Pngcs.Chunks; 
using System; 
using System.Collections.Generic; 
using System.Diagnostics; 
using System.IO; 
using System.Linq; 
using System.Text; 
using System.Threading.Tasks; 

namespace MarkerGenerator.Utils 
{ 
    class PngUtils 
    { 

     public string getMetadata(string file, string key) 
     { 

      PngReader pngr = FileHelper.CreatePngReader(file); 
      //pngr.MaxTotalBytesRead = 1024 * 1024 * 1024L * 3; // 3Gb! 
      //pngr.ReadSkippingAllRows(); 
      string data = pngr.GetMetadata().GetTxtForKey(key); 
      pngr.End(); 
      return data; ; 
     } 


     public static void addMetadata(String origFilename, Dictionary<string, string> data) 
     { 
      String destFilename = "tmp.png"; 
      PngReader pngr = FileHelper.CreatePngReader(origFilename); // or you can use the constructor 
      PngWriter pngw = FileHelper.CreatePngWriter(destFilename, pngr.ImgInfo, true); // idem 
      //Console.WriteLine(pngr.ToString()); // just information 
      int chunkBehav = ChunkCopyBehaviour.COPY_ALL_SAFE; // tell to copy all 'safe' chunks 
      pngw.CopyChunksFirst(pngr, chunkBehav);   // copy some metadata from reader 
      foreach (string key in data.Keys) 
      { 
       PngChunk chunk = pngw.GetMetadata().SetText(key, data[key]); 
       chunk.Priority = true; 
      } 

      int channels = pngr.ImgInfo.Channels; 
      if (channels < 3) 
       throw new Exception("This example works only with RGB/RGBA images"); 
      for (int row = 0; row < pngr.ImgInfo.Rows; row++) 
      { 
       ImageLine l1 = pngr.ReadRowInt(row); // format: RGBRGB... or RGBARGBA... 
       pngw.WriteRow(l1, row); 
      } 
      pngw.CopyChunksLast(pngr, chunkBehav); // metadata after the image pixels? can happen 
      pngw.End(); // dont forget this 
      pngr.End(); 
      File.Delete(origFilename); 
      File.Move(destFilename, origFilename); 

     } 

     public static void addMetadata(String origFilename,string key,string value) 
     { 
      Dictionary<string, string> data = new Dictionary<string, string>(); 
      data.Add(key, value); 
      addMetadata(origFilename, data); 
     } 


    } 
}