1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 03:33:20 +08:00
osu-lazer/osu.Game.Rulesets.Osu/UI/ReplayAnalysisOverlay.cs

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

159 lines
5.9 KiB
C#
Raw Normal View History

2024-09-03 16:49:50 +08: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.
2024-09-05 15:01:28 +08:00
using System.Threading;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
2024-09-05 15:01:28 +08:00
using osu.Framework.Caching;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Replays;
using osu.Game.Rulesets.Osu.Configuration;
using osu.Game.Rulesets.Osu.Replays;
using osu.Game.Rulesets.Osu.UI.ReplayAnalysis;
namespace osu.Game.Rulesets.Osu.UI
{
2024-09-04 17:43:33 +08:00
public partial class ReplayAnalysisOverlay : CompositeDrawable
{
private BindableBool showClickMarkers { get; } = new BindableBool();
private BindableBool showFrameMarkers { get; } = new BindableBool();
private BindableBool showCursorPath { get; } = new BindableBool();
private BindableInt displayLength { get; } = new BindableInt();
2024-09-05 15:01:28 +08:00
protected ClickMarkerContainer? ClickMarkers;
protected FrameMarkerContainer? FrameMarkers;
protected CursorPathContainer? CursorPath;
private readonly Replay replay;
2024-09-06 17:41:07 +08:00
private int replayFrameIndex;
public ReplayAnalysisOverlay(Replay replay)
{
RelativeSizeAxes = Axes.Both;
this.replay = replay;
}
private bool requireDisplay => showClickMarkers.Value || showFrameMarkers.Value || showCursorPath.Value;
[BackgroundDependencyLoader]
private void load(OsuRulesetConfigManager config)
{
config.BindWith(OsuRulesetSetting.ReplayClickMarkersEnabled, showClickMarkers);
config.BindWith(OsuRulesetSetting.ReplayFrameMarkersEnabled, showFrameMarkers);
config.BindWith(OsuRulesetSetting.ReplayCursorPathEnabled, showCursorPath);
config.BindWith(OsuRulesetSetting.ReplayAnalysisDisplayLength, displayLength);
}
protected override void LoadComplete()
{
base.LoadComplete();
displayLength.BindValueChanged(_ =>
{
2024-09-05 15:01:28 +08:00
// Need to fully reload to make this work.
invalidateLoaded();
}, true);
}
2024-09-06 17:41:07 +08:00
/// <summary>
2024-09-09 14:06:24 +08:00
/// Invalidated when containers are not loaded nor loading, false if loading, and true if loaded
2024-09-06 17:41:07 +08:00
/// </summary>
2024-09-09 14:06:24 +08:00
/// <remarks>
/// Knowing the loading/loaded state is for avoiding an enumeration error when adding
/// new entries and not starting a new load while loading
/// </remarks>
2024-09-06 17:41:07 +08:00
private readonly Cached<bool> loadState = new Cached<bool>();
2024-09-05 15:01:28 +08:00
private CancellationTokenSource? generationCancellationSource;
private void invalidateLoaded()
{
2024-09-06 17:41:07 +08:00
loadState.Invalidate();
replayFrameIndex = 0;
}
2024-09-05 15:01:28 +08:00
protected override void Update()
{
base.Update();
if (requireDisplay)
2024-09-06 17:41:07 +08:00
{
2024-09-05 15:01:28 +08:00
initialise();
2024-09-06 17:41:07 +08:00
// adding entries while the component is asynchronously loading
// can collide with enumeration operations and cause an error
if (loadState.IsValid && loadState.Value)
addEntries();
}
2024-09-05 15:01:28 +08:00
2024-09-05 15:45:37 +08:00
if (ClickMarkers != null) ClickMarkers.Alpha = showClickMarkers.Value ? 1 : 0;
if (FrameMarkers != null) FrameMarkers.Alpha = showFrameMarkers.Value ? 1 : 0;
if (CursorPath != null) CursorPath.Alpha = showCursorPath.Value ? 1 : 0;
2024-09-05 15:01:28 +08:00
}
private void initialise()
{
2024-09-06 17:41:07 +08:00
if (loadState.IsValid)
return;
2024-09-06 17:41:07 +08:00
loadState.Value = false;
2024-09-05 15:01:28 +08:00
generationCancellationSource?.Cancel();
generationCancellationSource = new CancellationTokenSource();
// It's faster to reinitialise the whole drawable stack than use `Clear` on `PooledDrawableWithLifetimeContainer`
2024-09-05 15:01:28 +08:00
var newDrawables = new Drawable[]
{
CursorPath = new CursorPathContainer(),
ClickMarkers = new ClickMarkerContainer(),
FrameMarkers = new FrameMarkerContainer(),
};
2024-09-06 17:41:07 +08:00
LoadComponentsAsync(newDrawables, drawables =>
{
InternalChildrenEnumerable = drawables;
loadState.Value = true;
}, generationCancellationSource.Token);
}
private void addEntries()
{
bool leftHeld = false;
bool rightHeld = false;
2024-09-05 15:01:28 +08:00
// This should probably be async as well, but it's a bit of a pain to debounce and everything.
// Let's address concerns when they are raised.
while (replayFrameIndex < replay.Frames.Count)
{
var osuFrame = (OsuReplayFrame)replay.Frames[replayFrameIndex];
bool leftButton = osuFrame.Actions.Contains(OsuAction.LeftButton);
bool rightButton = osuFrame.Actions.Contains(OsuAction.RightButton);
if (leftHeld && !leftButton)
leftHeld = false;
else if (!leftHeld && leftButton)
{
leftHeld = true;
ClickMarkers!.Add(new AnalysisFrameEntry(osuFrame.Time, displayLength.Value, osuFrame.Position, OsuAction.LeftButton));
}
if (rightHeld && !rightButton)
rightHeld = false;
else if (!rightHeld && rightButton)
{
rightHeld = true;
ClickMarkers!.Add(new AnalysisFrameEntry(osuFrame.Time, displayLength.Value, osuFrame.Position, OsuAction.RightButton));
}
2024-09-04 20:04:59 +08:00
FrameMarkers!.Add(new AnalysisFrameEntry(osuFrame.Time, displayLength.Value, osuFrame.Position, osuFrame.Actions.ToArray()));
CursorPath!.Add(new AnalysisFrameEntry(osuFrame.Time, displayLength.Value, osuFrame.Position));
2024-09-05 15:01:28 +08:00
replayFrameIndex++;
}
}
}
}