mirror of
https://github.com/ppy/osu.git
synced 2025-01-18 21:02:56 +08:00
124 lines
5.1 KiB
C#
124 lines
5.1 KiB
C#
// 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;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using osu.Game.Beatmaps;
|
|
using osu.Game.Rulesets.Mods;
|
|
using osu.Game.Storyboards.Drawables;
|
|
using osu.Game.Utils;
|
|
|
|
namespace osu.Game.Storyboards
|
|
{
|
|
public class Storyboard
|
|
{
|
|
private readonly Dictionary<string, StoryboardLayer> layers = new Dictionary<string, StoryboardLayer>();
|
|
public IEnumerable<StoryboardLayer> Layers => layers.Values;
|
|
|
|
public BeatmapInfo BeatmapInfo = new BeatmapInfo();
|
|
public IBeatmap Beatmap { get; set; } = new Beatmap();
|
|
|
|
/// <summary>
|
|
/// Whether the storyboard should prefer textures from the current skin before using local storyboard textures.
|
|
/// </summary>
|
|
public bool UseSkinSprites { get; set; }
|
|
|
|
public bool HasDrawable => Layers.Any(l => l.Elements.Any(e => e.IsDrawable));
|
|
|
|
/// <summary>
|
|
/// Across all layers, find the earliest point in time that a storyboard element exists at.
|
|
/// Will return null if there are no elements.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This iterates all elements and as such should be used sparingly or stored locally.
|
|
/// Sample events use their start time as "end time" during this calculation.
|
|
/// Video and background events are not included to match stable.
|
|
/// </remarks>
|
|
public double? EarliestEventTime => Layers.SelectMany(l => l.Elements)
|
|
.Where(e => e is not StoryboardVideo)
|
|
.MinBy(e => e.StartTime)?.StartTime;
|
|
|
|
/// <summary>
|
|
/// Across all layers, find the latest point in time that a storyboard element ends at.
|
|
/// Will return null if there are no elements.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This iterates all elements and as such should be used sparingly or stored locally.
|
|
/// Sample events use their start time as "end time" during this calculation.
|
|
/// Video and background events are not included to match stable.
|
|
/// </remarks>
|
|
public double? LatestEventTime => Layers.SelectMany(l => l.Elements)
|
|
.Where(e => e is not StoryboardVideo)
|
|
.MaxBy(e => e.GetEndTime())?.GetEndTime();
|
|
|
|
/// <summary>
|
|
/// Depth of the currently front-most storyboard layer, excluding the overlay layer.
|
|
/// </summary>
|
|
private int minimumLayerDepth;
|
|
|
|
public Storyboard()
|
|
{
|
|
layers.Add("Video", new StoryboardVideoLayer("Video", 4, false));
|
|
layers.Add("Background", new StoryboardLayer("Background", 3));
|
|
layers.Add("Fail", new StoryboardLayer("Fail", 2) { VisibleWhenPassing = false, });
|
|
layers.Add("Pass", new StoryboardLayer("Pass", 1) { VisibleWhenFailing = false, });
|
|
layers.Add("Foreground", new StoryboardLayer("Foreground", minimumLayerDepth = 0));
|
|
|
|
layers.Add("Overlay", new StoryboardLayer("Overlay", int.MinValue));
|
|
}
|
|
|
|
public StoryboardLayer GetLayer(string name)
|
|
{
|
|
if (!layers.TryGetValue(name, out var layer))
|
|
layers[name] = layer = new StoryboardLayer(name, --minimumLayerDepth);
|
|
|
|
return layer;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Whether the beatmap's background should be hidden while this storyboard is being displayed.
|
|
/// </summary>
|
|
public bool ReplacesBackground
|
|
{
|
|
get
|
|
{
|
|
string backgroundPath = BeatmapInfo.Metadata.BackgroundFile;
|
|
|
|
if (string.IsNullOrEmpty(backgroundPath))
|
|
return false;
|
|
|
|
// Importantly, do this after the NullOrEmpty because EF may have stored the non-nullable value as null to the database, bypassing compile-time constraints.
|
|
backgroundPath = backgroundPath.ToLowerInvariant();
|
|
|
|
return GetLayer("Background").Elements.Any(e => string.Equals(e.Path, backgroundPath, StringComparison.OrdinalIgnoreCase));
|
|
}
|
|
}
|
|
|
|
public virtual DrawableStoryboard CreateDrawable(IReadOnlyList<Mod>? mods = null) =>
|
|
new DrawableStoryboard(this, mods);
|
|
|
|
public virtual string? GetStoragePathFromStoryboardPath(string path)
|
|
{
|
|
string? resolvedPath = null;
|
|
|
|
if (Path.HasExtension(path))
|
|
{
|
|
resolvedPath = BeatmapInfo.BeatmapSet?.GetPathForFile(path);
|
|
}
|
|
else
|
|
{
|
|
// Some old storyboards don't include a file extension, so let's best guess at one.
|
|
foreach (string ext in SupportedExtensions.IMAGE_EXTENSIONS)
|
|
{
|
|
if ((resolvedPath = BeatmapInfo.BeatmapSet?.GetPathForFile($"{path}{ext}")) != null)
|
|
break;
|
|
}
|
|
}
|
|
|
|
return resolvedPath;
|
|
}
|
|
}
|
|
}
|