mirror of
https://github.com/ppy/osu.git
synced 2025-01-24 06:33:20 +08:00
67ca7e4135
Closes https://github.com/ppy/osu/issues/6842. This is a rather barebones implementation, just to get this in place somehow at least. The logic is simple - 50% health or above shows pass layer, anything below shows fail layer. This does not match stable logic all across the board because I have no idea how to package that. Stable defines "passing" in like fifty ways: - in mania it's >80% HP (bb57924c15/osu
!/GameModes/Play/Rulesets/Mania/RulesetMania.cs#L333-L336) - in taiko it's >80% *accuracy* (bb57924c15/osu
!/GameModes/Play/Rulesets/Taiko/RulesetTaiko.cs#L486-L492) - there's also the part where "geki additions" will unconditionally set passing state (bb57924c15/osu
!/GameModes/Play/Player.cs#L3561-L3564) - and also the part where at the end of the map, the final passing state is determined by checking whether the user passed more sections than failed (bb57924c15/osu
!/GameModes/Play/Player.cs#L3320) The biggest issues of these are probably the first two, and they can *probably* be fixed, but would require a new member on `Ruleset` and I'm not sure how to make one look, so I'm not doing that at this time pending collection of ideas on how to do that.
176 lines
6.1 KiB
C#
176 lines
6.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 System.Threading;
|
|
using System.Threading.Tasks;
|
|
using osu.Framework.Allocation;
|
|
using osu.Framework.Bindables;
|
|
using osu.Framework.Graphics;
|
|
using osu.Framework.Graphics.Containers;
|
|
using osu.Framework.Graphics.Textures;
|
|
using osu.Framework.IO.Stores;
|
|
using osu.Framework.Platform;
|
|
using osu.Game.Database;
|
|
using osu.Game.Rulesets.Mods;
|
|
using osu.Game.Screens.Play;
|
|
using osuTK;
|
|
|
|
namespace osu.Game.Storyboards.Drawables
|
|
{
|
|
public partial class DrawableStoryboard : Container<DrawableStoryboardLayer>
|
|
{
|
|
[Cached(typeof(Storyboard))]
|
|
public Storyboard Storyboard { get; }
|
|
|
|
/// <summary>
|
|
/// Whether the storyboard is considered finished.
|
|
/// </summary>
|
|
public IBindable<bool> HasStoryboardEnded => hasStoryboardEnded;
|
|
|
|
private readonly BindableBool hasStoryboardEnded = new BindableBool(true);
|
|
|
|
protected override Container<DrawableStoryboardLayer> Content { get; }
|
|
|
|
protected override Vector2 DrawScale => new Vector2(Parent!.DrawHeight / 480);
|
|
|
|
public override bool RemoveCompletedTransforms => false;
|
|
|
|
private double? lastEventEndTime;
|
|
|
|
[Cached(typeof(IReadOnlyList<Mod>))]
|
|
public IReadOnlyList<Mod> Mods { get; }
|
|
|
|
[Resolved]
|
|
private GameHost host { get; set; } = null!;
|
|
|
|
[Resolved]
|
|
private RealmAccess realm { get; set; } = null!;
|
|
|
|
private DependencyContainer dependencies = null!;
|
|
|
|
private BindableNumber<double> health = null!;
|
|
private readonly BindableBool passing = new BindableBool(true);
|
|
|
|
protected override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent) =>
|
|
dependencies = new DependencyContainer(base.CreateChildDependencies(parent));
|
|
|
|
public DrawableStoryboard(Storyboard storyboard, IReadOnlyList<Mod>? mods = null)
|
|
{
|
|
Storyboard = storyboard;
|
|
Mods = mods ?? Array.Empty<Mod>();
|
|
|
|
Size = new Vector2(640, 480);
|
|
|
|
bool onlyHasVideoElements = Storyboard.Layers.SelectMany(l => l.Elements).All(e => e is StoryboardVideo);
|
|
|
|
Width = Height * (storyboard.BeatmapInfo.WidescreenStoryboard || onlyHasVideoElements ? 16 / 9f : 4 / 3f);
|
|
|
|
Anchor = Anchor.Centre;
|
|
Origin = Anchor.Centre;
|
|
|
|
AddInternal(Content = new Container<DrawableStoryboardLayer>
|
|
{
|
|
RelativeSizeAxes = Axes.Both,
|
|
Anchor = Anchor.Centre,
|
|
Origin = Anchor.Centre,
|
|
});
|
|
}
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(IGameplayClock? clock, CancellationToken? cancellationToken, GameplayState? gameplayState)
|
|
{
|
|
if (clock != null)
|
|
Clock = clock;
|
|
|
|
dependencies.CacheAs(typeof(TextureStore),
|
|
new TextureStore(host.Renderer, host.CreateTextureLoaderStore(
|
|
CreateResourceLookupStore()
|
|
), false, scaleAdjust: 1));
|
|
|
|
foreach (var layer in Storyboard.Layers)
|
|
{
|
|
cancellationToken?.ThrowIfCancellationRequested();
|
|
|
|
Add(layer.CreateDrawable());
|
|
}
|
|
|
|
lastEventEndTime = Storyboard.LatestEventTime;
|
|
|
|
health = gameplayState?.HealthProcessor.Health.GetBoundCopy() ?? new BindableDouble(1);
|
|
}
|
|
|
|
protected override void LoadComplete()
|
|
{
|
|
base.LoadComplete();
|
|
|
|
health.BindValueChanged(val => passing.Value = val.NewValue >= 0.5, true);
|
|
passing.BindValueChanged(_ => updateLayerVisibility(), true);
|
|
}
|
|
|
|
protected virtual IResourceStore<byte[]> CreateResourceLookupStore() => new StoryboardResourceLookupStore(Storyboard, realm, host);
|
|
|
|
protected override void Update()
|
|
{
|
|
base.Update();
|
|
hasStoryboardEnded.Value = lastEventEndTime == null || Time.Current >= lastEventEndTime;
|
|
}
|
|
|
|
public DrawableStoryboardLayer OverlayLayer => Children.Single(layer => layer.Name == "Overlay");
|
|
|
|
private void updateLayerVisibility()
|
|
{
|
|
foreach (var layer in Children)
|
|
layer.Enabled = passing.Value ? layer.Layer.VisibleWhenPassing : layer.Layer.VisibleWhenFailing;
|
|
}
|
|
|
|
private class StoryboardResourceLookupStore : IResourceStore<byte[]>
|
|
{
|
|
private readonly IResourceStore<byte[]> realmFileStore;
|
|
private readonly Storyboard storyboard;
|
|
|
|
public StoryboardResourceLookupStore(Storyboard storyboard, RealmAccess realm, GameHost host)
|
|
{
|
|
realmFileStore = new RealmFileStore(realm, host.Storage).Store;
|
|
this.storyboard = storyboard;
|
|
}
|
|
|
|
public void Dispose() =>
|
|
realmFileStore.Dispose();
|
|
|
|
public byte[] Get(string name)
|
|
{
|
|
string? storagePath = storyboard.GetStoragePathFromStoryboardPath(name);
|
|
|
|
return string.IsNullOrEmpty(storagePath)
|
|
? null!
|
|
: realmFileStore.Get(storagePath);
|
|
}
|
|
|
|
public Task<byte[]> GetAsync(string name, CancellationToken cancellationToken = new CancellationToken())
|
|
{
|
|
string? storagePath = storyboard.GetStoragePathFromStoryboardPath(name);
|
|
|
|
return string.IsNullOrEmpty(storagePath)
|
|
? Task.FromResult<byte[]>(null!)
|
|
: realmFileStore.GetAsync(storagePath, cancellationToken);
|
|
}
|
|
|
|
public Stream? GetStream(string name)
|
|
{
|
|
string? storagePath = storyboard.GetStoragePathFromStoryboardPath(name);
|
|
|
|
return string.IsNullOrEmpty(storagePath)
|
|
? null
|
|
: realmFileStore.GetStream(storagePath);
|
|
}
|
|
|
|
public IEnumerable<string> GetAvailableResources() =>
|
|
realmFileStore.GetAvailableResources();
|
|
}
|
|
}
|
|
}
|