mirror of
https://github.com/rosenbjerg/FFMpegCore.git
synced 2025-12-16 02:55:44 +00:00
Compare commits
1 commit
2eecb92481
...
e35aadb0f0
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e35aadb0f0 |
3 changed files with 0 additions and 101 deletions
|
|
@ -82,40 +82,6 @@ public class VideoTest
|
||||||
Assert.IsTrue(success);
|
Assert.IsTrue(success);
|
||||||
}
|
}
|
||||||
|
|
||||||
[TestMethod]
|
|
||||||
[Timeout(BaseTimeoutMilliseconds, CooperativeCancellation = true)]
|
|
||||||
public async Task Video_MetadataBuilder()
|
|
||||||
{
|
|
||||||
using var outputFile = new TemporaryFile($"out{VideoType.Mp4.Extension}");
|
|
||||||
|
|
||||||
await FFMpegArguments
|
|
||||||
.FromFileInput(TestResources.WebmVideo)
|
|
||||||
.AddMetaData(FFMetadataBuilder.Empty()
|
|
||||||
.WithTag("title", "noname")
|
|
||||||
.WithTag("artist", "unknown")
|
|
||||||
.WithChapter("Chapter 1", 1.1)
|
|
||||||
.WithChapter("Chapter 2", 1.23))
|
|
||||||
.OutputToFile(outputFile, false, opt => opt
|
|
||||||
.WithVideoCodec(VideoCodec.LibX264))
|
|
||||||
.CancellableThrough(TestContext.CancellationToken)
|
|
||||||
.ProcessAsynchronously();
|
|
||||||
|
|
||||||
var analysis = await FFProbe.AnalyseAsync(outputFile, cancellationToken: TestContext.CancellationToken);
|
|
||||||
Assert.IsTrue(analysis.Format.Tags!.TryGetValue("title", out var title));
|
|
||||||
Assert.IsTrue(analysis.Format.Tags!.TryGetValue("artist", out var artist));
|
|
||||||
Assert.AreEqual("noname", title);
|
|
||||||
Assert.AreEqual("unknown", artist);
|
|
||||||
|
|
||||||
Assert.HasCount(2, analysis.Chapters);
|
|
||||||
Assert.AreEqual("Chapter 1", analysis.Chapters.First().Title);
|
|
||||||
Assert.AreEqual(1.1, analysis.Chapters.First().Duration.TotalSeconds);
|
|
||||||
Assert.AreEqual(1.1, analysis.Chapters.First().End.TotalSeconds);
|
|
||||||
|
|
||||||
Assert.AreEqual("Chapter 2", analysis.Chapters.Last().Title);
|
|
||||||
Assert.AreEqual(1.23, analysis.Chapters.Last().Duration.TotalSeconds);
|
|
||||||
Assert.AreEqual(1.1 + 1.23, analysis.Chapters.Last().End.TotalSeconds);
|
|
||||||
}
|
|
||||||
|
|
||||||
[TestMethod]
|
[TestMethod]
|
||||||
[Timeout(BaseTimeoutMilliseconds, CooperativeCancellation = true)]
|
[Timeout(BaseTimeoutMilliseconds, CooperativeCancellation = true)]
|
||||||
public void Video_ToH265_MKV_Args()
|
public void Video_ToH265_MKV_Args()
|
||||||
|
|
|
||||||
|
|
@ -1,62 +0,0 @@
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
namespace FFMpegCore;
|
|
||||||
|
|
||||||
public class FFMetadataBuilder
|
|
||||||
{
|
|
||||||
private Dictionary<string, string> Tags { get; } = new();
|
|
||||||
private List<FFMetadataChapter> Chapters { get; } = [];
|
|
||||||
|
|
||||||
public static FFMetadataBuilder Empty()
|
|
||||||
{
|
|
||||||
return new FFMetadataBuilder();
|
|
||||||
}
|
|
||||||
|
|
||||||
public FFMetadataBuilder WithTag(string key, string value)
|
|
||||||
{
|
|
||||||
Tags.Add(key, value);
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public FFMetadataBuilder WithChapter(string title, long durationMs)
|
|
||||||
{
|
|
||||||
Chapters.Add(new FFMetadataChapter(title, durationMs));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public FFMetadataBuilder WithChapter(string title, double durationSeconds)
|
|
||||||
{
|
|
||||||
Chapters.Add(new FFMetadataChapter(title, Convert.ToInt64(durationSeconds * 1000)));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetMetadataFileContent()
|
|
||||||
{
|
|
||||||
var sb = new StringBuilder();
|
|
||||||
sb.AppendLine(";FFMETADATA1");
|
|
||||||
|
|
||||||
foreach (var tag in Tags)
|
|
||||||
{
|
|
||||||
sb.AppendLine($"{tag.Key}={tag.Value}");
|
|
||||||
}
|
|
||||||
|
|
||||||
long totalDurationMs = 0;
|
|
||||||
foreach (var chapter in Chapters)
|
|
||||||
{
|
|
||||||
sb.AppendLine("[CHAPTER]");
|
|
||||||
sb.AppendLine("TIMEBASE=1/1000");
|
|
||||||
sb.AppendLine($"START={totalDurationMs}");
|
|
||||||
sb.AppendLine($"END={totalDurationMs + chapter.DurationMs}");
|
|
||||||
sb.AppendLine($"title={chapter.Title}");
|
|
||||||
totalDurationMs += chapter.DurationMs;
|
|
||||||
}
|
|
||||||
|
|
||||||
return sb.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private class FFMetadataChapter(string title, long durationMs)
|
|
||||||
{
|
|
||||||
public string Title { get; } = title;
|
|
||||||
public long DurationMs { get; } = durationMs;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -109,11 +109,6 @@ public sealed class FFMpegArguments : FFMpegArgumentsBase
|
||||||
return WithInput(new MetaDataArgument(content), addArguments);
|
return WithInput(new MetaDataArgument(content), addArguments);
|
||||||
}
|
}
|
||||||
|
|
||||||
public FFMpegArguments AddMetaData(FFMetadataBuilder metaDataBuilder, Action<FFMpegArgumentOptions>? addArguments = null)
|
|
||||||
{
|
|
||||||
return WithInput(new MetaDataArgument(metaDataBuilder.GetMetadataFileContent()), addArguments);
|
|
||||||
}
|
|
||||||
|
|
||||||
public FFMpegArguments AddMetaData(IReadOnlyMetaData metaData, Action<FFMpegArgumentOptions>? addArguments = null)
|
public FFMpegArguments AddMetaData(IReadOnlyMetaData metaData, Action<FFMpegArgumentOptions>? addArguments = null)
|
||||||
{
|
{
|
||||||
return WithInput(new MetaDataArgument(MetaDataSerializer.Instance.Serialize(metaData)), addArguments);
|
return WithInput(new MetaDataArgument(MetaDataSerializer.Instance.Serialize(metaData)), addArguments);
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue