2012-04-02 19 views
4

Ich habe viele MPEG-4-Dateien transcodiert von verschiedenen Digitalkamera-Formaten, für die das Dateisystem-Änderungsdatum korrekt ist. Ich möchte den Tag "Media Created" passend einstellen. Dies kann manuell im Windows Explorer über die Registerkarte Details des Fensters Eigenschaften vorgenommen werden. Das Festlegen von Medien ist nützlich, da Windows Live Photo Gallery dieses Feld für die Eigenschaft "Datum genommen" deaktiviert. Leider macht die Anzahl der Dateien die manuelle Einstellung all ihrer Daten unpraktisch.So legen Sie das "Media created" -Datum in einer MPEG-4-Datei fest

Ein paar Möglichkeiten für die Automatisierung haben Potenzial. TagLib# scheint alle MP4-Tags zu unterstützen, aber die API, um mehr als einfache Tags zu bekommen, ist unklar. Ein anderer Blickwinkel ist die Windows-Shell. Vermutlich verwendet Windows Explorer es, um die Tags zu schreiben. Es gibt eine example for reading über die Shell, aber es scheint keine API zum Schreiben zu geben.

Antwort

1

Ich löste dies durch Lesen Schreiben/Schreiben des MP4-Dateiformat direkt. Hier ist der Code in VB:

Sub Main() 
    ' Retrieve creation-time and modification-time, embedded inside the metadata of MP4 files 
    Dim ft = Mp4Times("a.mp4") 
    Console.WriteLine(ft.CreationTime) 
    Console.WriteLine(ft.ModificationTime) 

    ' Update those times 
    Mp4Times("a.mp4", Date.Now, Date.Now) 
End Sub 

Class FileTimes 
    Public CreationTime As Date 
    Public ModificationTime As Date 
End Class 

Function Mp4Times(fn As String, Optional newCreationTime As Date? = Nothing, Optional newModificationTime As Date? = Nothing) As FileTimes 
    Dim ft As FileTimes 
    Using f = If(newCreationTime.HasValue OrElse newModificationTime.HasValue, IO.File.Open(fn, IO.FileMode.Open), IO.File.OpenRead(fn)) 
     f.Seek(0, IO.SeekOrigin.End) : Dim fend = f.Position 

     ' The file is made up of a sequence of boxes, with a standard way to find size and FourCC "kind" of each. 
     ' Some box kinds contain a kind-specific blob of binary data. Other box kinds contain a sequence 
     ' of sub-boxes. You need to look up the specs for each kind to know whether it has a blob or sub-boxes. 
     ' We look for a top-level box of kind "moov", which contains sub-boxes, and then we look for its sub-box 
     ' of kind "mvhd", which contains a binary blob. This is where Creation/ModificationTime are stored. 
     Dim pos = 0L, payloadStart = 0L, payloadEnd = 0L, boxKind = "" 
     While ReadNextBoxInfo(f, pos, fend, boxKind, payloadStart, payloadEnd) AndAlso boxKind <> "moov" 
      pos = payloadEnd 
     End While 
     If boxKind <> "moov" Then Return Nothing 
     pos = payloadStart : fend = payloadEnd 
     While ReadNextBoxInfo(f, pos, fend, boxKind, payloadStart, payloadEnd) AndAlso boxKind <> "mvhd" 
      pos = payloadEnd 
     End While 
     If boxKind <> "mvhd" Then Return Nothing 

     ' The "mvhd" binary blob consists of 1byte (version, either 0 or 1), 3bytes (flags), 
     ' and then either 4bytes (creation), 4bytes (modification) 
     ' or 8bytes (creation), 8bytes (modification) 
     ' If version=0 then it's the former, otherwise it's the later. 
     ' In both cases "creation" and "modification" are big-endian number of seconds since 1st Jan 1904 UTC 
     f.Seek(pos + 8, IO.SeekOrigin.Begin) : Dim version = f.ReadByte() 
     f.Seek(pos + 12, IO.SeekOrigin.Begin) 
     Dim creationTime As Date, modificationTime As Date 
     ' 
     If newCreationTime.HasValue Then 
      creationTime = newCreationTime.Value 
      If version = 0 Then Write4byteDate(f, creationTime) Else Write8byteDate(f, creationTime) 
     Else 
      creationTime = If(version = 0, ReadNext4byteDate(f), ReadNext8byteDate(f)) 
     End If 
     ' 
     If newModificationTime.HasValue Then 
      modificationTime = newModificationTime.Value 
      If version = 0 Then Write4byteDate(f, modificationTime) Else Write8byteDate(f, modificationTime) 
     Else 
      modificationTime = If(version = 0, ReadNext4byteDate(f), ReadNext8byteDate(f)) 
     End If 
     ft = New FileTimes With {.CreationTime = creationTime, .ModificationTime = modificationTime} 
    End Using 

    If newCreationTime.HasValue Then IO.File.SetCreationTime(fn, newCreationTime.Value) 
    If newModificationTime.HasValue Then IO.File.SetLastWriteTime(fn, newModificationTime.Value) 
    Return ft 
End Function 

Function ReadNextBoxInfo(f As IO.Stream, pos As Long, fend As Long, ByRef boxKind As String, ByRef payloadStart As Long, ByRef payloadEnd As Long) As Boolean 
    boxKind = "" : payloadStart = 0 : payloadEnd = 0 
    If pos + 8 > fend Then Return False 
    Dim b(3) As Byte 
    f.Seek(pos, IO.SeekOrigin.Begin) 
    f.Read(b, 0, 4) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
    Dim size = BitConverter.ToUInt32(b, 0) 
    f.Read(b, 0, 4) 
    Dim kind = ChrW(b(0)) & ChrW(b(1)) & ChrW(b(2)) & ChrW(b(3)) 
    If size <> 1 Then 
     If pos + size > fend Then Return False 
     boxKind = kind : payloadStart = pos + 8 : payloadEnd = payloadStart + size - 8 : Return True 
    End If 
    If size = 1 AndAlso pos + 16 <= fend Then 
     ReDim b(7) 
     f.Read(b, 0, 8) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
     Dim size2 = CLng(BitConverter.ToUInt64(b, 0)) 
     If pos + size2 > fend Then Return False 
     boxKind = kind : payloadStart = pos + 16 : payloadEnd = payloadStart + size2 - 16 : Return True 
    End If 
    Return False 
End Function 

ReadOnly TZERO As Date = New Date(1904, 1, 1, 0, 0, 0, DateTimeKind.Utc) 

Function ReadNext4byteDate(f As IO.Stream) As Date 
    Dim b(3) As Byte 
    f.Read(b, 0, 4) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
    Dim secs = BitConverter.ToUInt32(b, 0) 
    Return TZERO.AddSeconds(secs) 
End Function 

Function ReadNext8byteDate(f As IO.Stream) As Date 
    Dim b(7) As Byte 
    f.Read(b, 0, 8) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
    Dim secs = BitConverter.ToUInt64(b, 0) 
    Return TZERO.AddSeconds(secs) 
End Function 

Sub Write4byteDate(f As IO.Stream, d As Date) 
    Dim secs = CUInt((d - TZERO).TotalSeconds) 
    Dim b = BitConverter.GetBytes(secs) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
    f.Write(b, 0, 4) 
End Sub 

Sub Write8byteDate(f As IO.Stream, d As Date) 
    Dim secs = CULng((d - TZERO).TotalSeconds) 
    Dim b = BitConverter.GetBytes(secs) : If BitConverter.IsLittleEndian Then Array.Reverse(b) 
    f.Write(b, 0, 8) 
End Sub 
4

Ich hatte Erfolg mit exiftool. Hier sind die Befehle, die ich verwendet habe, um eine Datei zu korrigieren. Exiftool kann jedoch verwendet werden, um Metadaten in Stapeln zu korrigieren.

C:\>exiftool.exe -s test.mp4 

ExifToolVersion     : 10.01 
FileName      : test.mp4 
Directory      : . 
FileSize      : 14 MB 
FileModifyDate     : 2015:09:04 22:33:16+05:00 
FileAccessDate     : 2015:09:05 14:10:08+05:00 
FileCreateDate     : 2015:09:05 14:10:08+05:00 
FilePermissions     : rw-rw-rw- 
FileType      : MP4 
FileTypeExtension    : mp4 
MIMEType      : video/mp4 
MajorBrand      : MP4 Base Media v1 [IS0 14496-12:2003] 
MinorVersion     : 0.2.0 
CompatibleBrands    : isom, iso2, avc1, mp41 
MovieHeaderVersion    : 0 
CreateDate      : 0000:00:00 00:00:00 
ModifyDate      : 0000:00:00 00:00:00 
TimeScale      : 1000 
Duration      : 0:01:22 
-- snip -- 
TrackHeaderVersion    : 0 
TrackCreateDate     : 0000:00:00 00:00:00 
TrackModifyDate     : 0000:00:00 00:00:00 
-- snip -- 
MediaHeaderVersion    : 0 
MediaCreateDate     : 0000:00:00 00:00:00 
MediaModifyDate     : 0000:00:00 00:00:00 
-- snip -- 

C:\>exiftool.exe^
-Quicktime:CreateDate="2007-01-02 03:04:05"^
-Quicktime:ModifyDate="2007-01-02 03:04:05"^
    -TrackCreateDate="2007-01-02 03:04:05"^
    -TrackModifyDate="2007-01-02 03:04:05"^
    -MediaCreateDate="2007-01-02 03:04:05"^
    -MediaModifyDate="2007-01-02 03:04:05"^
test.mp4 

C:\>exiftool.exe -s test.mp4 

FileModifyDate     : 2015:09:05 14:20:54+05:00 
FileAccessDate     : 2015:09:05 14:20:54+05:00 
FileCreateDate     : 2015:09:05 14:10:08+05:00 
-- snip -- 
CreateDate      : 2007:01:02 03:04:05 
ModifyDate      : 2007:01:02 03:04:05 
-- snip -- 
TrackCreateDate     : 2007:01:02 03:04:05 
TrackModifyDate     : 2007:01:02 03:04:05 
-- snip -- 
MediaCreateDate     : 2007:01:02 03:04:05 
MediaModifyDate     : 2007:01:02 03:04:05 
-- snip --