osu/osu.Game/Rulesets/Edit/Checks/CheckAudioInVideo.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

121 lines
4.4 KiB
C#
Raw Normal View History

2021-07-13 02:17:41 +00:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
using System;
2021-07-13 02:17:41 +00:00
using System.Collections.Generic;
using System.IO;
using osu.Framework.Logging;
using osu.Game.Beatmaps;
2021-10-11 11:56:26 +00:00
using osu.Game.IO.FileAbstraction;
2021-07-13 02:17:41 +00:00
using osu.Game.Rulesets.Edit.Checks.Components;
using osu.Game.Storyboards;
using TagLib;
using File = TagLib.File;
2021-07-13 02:17:41 +00:00
namespace osu.Game.Rulesets.Edit.Checks
{
public class CheckAudioInVideo : ICheck
{
public CheckMetadata Metadata => new CheckMetadata(CheckCategory.Audio, "Audio track in video files");
public IEnumerable<IssueTemplate> PossibleTemplates => new IssueTemplate[]
{
new IssueTemplateHasAudioTrack(this),
2021-10-11 11:53:23 +00:00
new IssueTemplateMissingFile(this),
new IssueTemplateFileError(this)
2021-07-13 02:17:41 +00:00
};
public IEnumerable<Issue> Run(BeatmapVerifierContext context)
{
var beatmapSet = context.Beatmap.BeatmapInfo.BeatmapSet;
2021-10-11 11:55:50 +00:00
var videoPaths = new List<string>();
2021-07-13 02:17:41 +00:00
foreach (var layer in context.WorkingBeatmap.Storyboard.Layers)
{
foreach (var element in layer.Elements)
{
if (!(element is StoryboardVideo video))
continue;
2021-10-11 11:55:50 +00:00
// Ensures we don't check the same video file multiple times in case of multiple elements using it.
if (!videoPaths.Contains(video.Path))
videoPaths.Add(video.Path);
}
}
2021-07-13 02:17:41 +00:00
foreach (string filename in videoPaths)
2021-10-11 11:55:50 +00:00
{
string? storagePath = beatmapSet?.GetPathForFile(filename);
2021-07-13 02:17:41 +00:00
2021-10-11 11:55:50 +00:00
if (storagePath == null)
{
// There's an element in the storyboard that requires this resource, so it being missing is worth warning about.
yield return new IssueTemplateMissingFile(this).Create(filename);
2021-07-13 02:17:41 +00:00
2021-10-11 11:55:50 +00:00
continue;
2021-07-13 02:17:41 +00:00
}
2021-10-11 11:55:50 +00:00
Issue issue;
2021-10-11 11:56:26 +00:00
try
{
// We use TagLib here for platform invariance; BASS cannot detect audio presence on Linux.
using (Stream data = context.WorkingBeatmap.GetStream(storagePath))
using (File tagFile = File.Create(new StreamFileAbstraction(filename, data)))
{
if (tagFile.Properties.AudioChannels == 0)
continue;
}
issue = new IssueTemplateHasAudioTrack(this).Create(filename);
2021-10-11 11:56:26 +00:00
}
catch (CorruptFileException)
2021-10-11 11:56:26 +00:00
{
issue = new IssueTemplateFileError(this).Create(filename, "Corrupt file");
}
catch (UnsupportedFormatException)
{
issue = new IssueTemplateFileError(this).Create(filename, "Unsupported format");
2021-10-11 11:56:26 +00:00
}
catch (Exception ex)
{
issue = new IssueTemplateFileError(this).Create(filename, "Internal failure - see logs for more info");
Logger.Log($"Failed when running {nameof(CheckAudioInVideo)}: {ex}");
}
2021-10-11 11:56:26 +00:00
yield return issue;
2021-07-13 02:17:41 +00:00
}
}
public class IssueTemplateHasAudioTrack : IssueTemplate
{
public IssueTemplateHasAudioTrack(ICheck check)
: base(check, IssueType.Problem, "\"{0}\" has an audio track.")
{
}
public Issue Create(string filename) => new Issue(this, filename);
}
2021-10-11 11:53:23 +00:00
public class IssueTemplateFileError : IssueTemplate
2021-07-13 02:17:41 +00:00
{
2021-10-11 11:53:23 +00:00
public IssueTemplateFileError(ICheck check)
: base(check, IssueType.Error, "Could not check whether \"{0}\" has an audio track ({1}).")
2021-07-13 02:17:41 +00:00
{
}
2021-10-11 11:53:23 +00:00
public Issue Create(string filename, string errorReason) => new Issue(this, filename, errorReason);
2021-07-13 02:17:41 +00:00
}
public class IssueTemplateMissingFile : IssueTemplate
{
public IssueTemplateMissingFile(ICheck check)
: base(check, IssueType.Warning, "Could not check whether \"{0}\" has an audio track, because it is missing.")
{
}
public Issue Create(string filename) => new Issue(this, filename);
}
}
}