mirror of
https://github.com/osukey/osukey.git
synced 2025-08-07 08:33:55 +09:00
Merge pull request #19599 from ItsShamed/kps
Add clicks/second counter to HUD
This commit is contained in:
BIN
osu.Game.Tests/Resources/Archives/modified-default-20220818.osk
Normal file
BIN
osu.Game.Tests/Resources/Archives/modified-default-20220818.osk
Normal file
Binary file not shown.
@ -36,7 +36,9 @@ namespace osu.Game.Tests.Skins
|
|||||||
"Archives/modified-default-20220723.osk",
|
"Archives/modified-default-20220723.osk",
|
||||||
"Archives/modified-classic-20220723.osk",
|
"Archives/modified-classic-20220723.osk",
|
||||||
// Covers legacy song progress, UR counter, colour hit error metre.
|
// Covers legacy song progress, UR counter, colour hit error metre.
|
||||||
"Archives/modified-classic-20220801.osk"
|
"Archives/modified-classic-20220801.osk",
|
||||||
|
// Covers clicks/s counter
|
||||||
|
"Archives/modified-default-20220818.osk"
|
||||||
};
|
};
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@ -0,0 +1,130 @@
|
|||||||
|
// 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 NUnit.Framework;
|
||||||
|
using osu.Framework.Audio;
|
||||||
|
using osu.Framework.Bindables;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Testing;
|
||||||
|
using osu.Framework.Timing;
|
||||||
|
using osu.Game.Screens.Play;
|
||||||
|
using osu.Game.Screens.Play.HUD.ClicksPerSecond;
|
||||||
|
using osuTK;
|
||||||
|
|
||||||
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
|
{
|
||||||
|
public class TestSceneClicksPerSecondCalculator : OsuTestScene
|
||||||
|
{
|
||||||
|
private ClicksPerSecondCalculator calculator = null!;
|
||||||
|
|
||||||
|
private TestGameplayClock manualGameplayClock = null!;
|
||||||
|
|
||||||
|
[SetUpSteps]
|
||||||
|
public void SetUpSteps()
|
||||||
|
{
|
||||||
|
AddStep("create components", () =>
|
||||||
|
{
|
||||||
|
manualGameplayClock = new TestGameplayClock();
|
||||||
|
|
||||||
|
Child = new DependencyProvidingContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
CachedDependencies = new (Type, object)[] { (typeof(IGameplayClock), manualGameplayClock) },
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
calculator = new ClicksPerSecondCalculator(),
|
||||||
|
new DependencyProvidingContainer
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both,
|
||||||
|
CachedDependencies = new (Type, object)[] { (typeof(ClicksPerSecondCalculator), calculator) },
|
||||||
|
Child = new ClicksPerSecondCounter
|
||||||
|
{
|
||||||
|
Anchor = Anchor.Centre,
|
||||||
|
Origin = Anchor.Centre,
|
||||||
|
Scale = new Vector2(5),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestBasicConsistency()
|
||||||
|
{
|
||||||
|
seek(1000);
|
||||||
|
AddStep("add inputs in past", () => addInputs(new double[] { 0, 100, 200, 300, 400, 500, 600, 700, 800, 900 }));
|
||||||
|
checkClicksPerSecondValue(10);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestRateAdjustConsistency()
|
||||||
|
{
|
||||||
|
seek(1000);
|
||||||
|
AddStep("add inputs in past", () => addInputs(new double[] { 0, 100, 200, 300, 400, 500, 600, 700, 800, 900 }));
|
||||||
|
checkClicksPerSecondValue(10);
|
||||||
|
AddStep("set rate 0.5x", () => manualGameplayClock.TrueGameplayRate = 0.5);
|
||||||
|
checkClicksPerSecondValue(5);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestInputsDiscardedOnRewind()
|
||||||
|
{
|
||||||
|
seek(1000);
|
||||||
|
AddStep("add inputs in past", () => addInputs(new double[] { 0, 100, 200, 300, 400, 500, 600, 700, 800, 900 }));
|
||||||
|
checkClicksPerSecondValue(10);
|
||||||
|
seek(500);
|
||||||
|
checkClicksPerSecondValue(6);
|
||||||
|
seek(1000);
|
||||||
|
checkClicksPerSecondValue(6);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void checkClicksPerSecondValue(int i) => AddAssert("clicks/s is correct", () => calculator.Value, () => Is.EqualTo(i));
|
||||||
|
|
||||||
|
private void seekClockImmediately(double time) => manualGameplayClock.CurrentTime = time;
|
||||||
|
|
||||||
|
private void seek(double time) => AddStep($"Seek to {time}ms", () => seekClockImmediately(time));
|
||||||
|
|
||||||
|
private void addInputs(IEnumerable<double> inputs)
|
||||||
|
{
|
||||||
|
double baseTime = manualGameplayClock.CurrentTime;
|
||||||
|
|
||||||
|
foreach (double timestamp in inputs)
|
||||||
|
{
|
||||||
|
seekClockImmediately(timestamp);
|
||||||
|
calculator.AddInputTimestamp();
|
||||||
|
}
|
||||||
|
|
||||||
|
seekClockImmediately(baseTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class TestGameplayClock : IGameplayClock
|
||||||
|
{
|
||||||
|
public double CurrentTime { get; set; }
|
||||||
|
|
||||||
|
public double Rate => 1;
|
||||||
|
|
||||||
|
public bool IsRunning => true;
|
||||||
|
|
||||||
|
public double TrueGameplayRate { set => adjustableAudioComponent.Tempo.Value = value; }
|
||||||
|
|
||||||
|
private readonly AudioAdjustments adjustableAudioComponent = new AudioAdjustments();
|
||||||
|
|
||||||
|
public void ProcessFrame()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public double ElapsedFrameTime => throw new NotImplementedException();
|
||||||
|
public double FramesPerSecond => throw new NotImplementedException();
|
||||||
|
public FrameTimeInfo TimeInfo => throw new NotImplementedException();
|
||||||
|
public double StartTime => throw new NotImplementedException();
|
||||||
|
|
||||||
|
public IAdjustableAudioComponent AdjustmentsFromMods => adjustableAudioComponent;
|
||||||
|
|
||||||
|
public IEnumerable<double> NonGameplayAdjustments => throw new NotImplementedException();
|
||||||
|
public IBindable<bool> IsPaused => throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -30,6 +30,7 @@ using osu.Game.Rulesets.Objects.Drawables;
|
|||||||
using osu.Game.Rulesets.Scoring;
|
using osu.Game.Rulesets.Scoring;
|
||||||
using osu.Game.Scoring;
|
using osu.Game.Scoring;
|
||||||
using osu.Game.Screens.Play;
|
using osu.Game.Screens.Play;
|
||||||
|
using osu.Game.Screens.Play.HUD.ClicksPerSecond;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.UI
|
namespace osu.Game.Rulesets.UI
|
||||||
@ -38,7 +39,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
/// Displays an interactive ruleset gameplay instance.
|
/// Displays an interactive ruleset gameplay instance.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <typeparam name="TObject">The type of HitObject contained by this DrawableRuleset.</typeparam>
|
/// <typeparam name="TObject">The type of HitObject contained by this DrawableRuleset.</typeparam>
|
||||||
public abstract class DrawableRuleset<TObject> : DrawableRuleset, IProvideCursor, ICanAttachKeyCounter
|
public abstract class DrawableRuleset<TObject> : DrawableRuleset, IProvideCursor, ICanAttachHUDPieces
|
||||||
where TObject : HitObject
|
where TObject : HitObject
|
||||||
{
|
{
|
||||||
public override event Action<JudgementResult> NewResult;
|
public override event Action<JudgementResult> NewResult;
|
||||||
@ -338,7 +339,10 @@ namespace osu.Game.Rulesets.UI
|
|||||||
public abstract DrawableHitObject<TObject> CreateDrawableRepresentation(TObject h);
|
public abstract DrawableHitObject<TObject> CreateDrawableRepresentation(TObject h);
|
||||||
|
|
||||||
public void Attach(KeyCounterDisplay keyCounter) =>
|
public void Attach(KeyCounterDisplay keyCounter) =>
|
||||||
(KeyBindingInputManager as ICanAttachKeyCounter)?.Attach(keyCounter);
|
(KeyBindingInputManager as ICanAttachHUDPieces)?.Attach(keyCounter);
|
||||||
|
|
||||||
|
public void Attach(ClicksPerSecondCalculator calculator) =>
|
||||||
|
(KeyBindingInputManager as ICanAttachHUDPieces)?.Attach(calculator);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates a key conversion input manager. An exception will be thrown if a valid <see cref="RulesetInputManager{T}"/> is not returned.
|
/// Creates a key conversion input manager. An exception will be thrown if a valid <see cref="RulesetInputManager{T}"/> is not returned.
|
||||||
|
@ -20,7 +20,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
[Cached(typeof(IGameplayClock))]
|
[Cached(typeof(IGameplayClock))]
|
||||||
[Cached(typeof(IFrameStableClock))]
|
[Cached(typeof(IFrameStableClock))]
|
||||||
public sealed class FrameStabilityContainer : Container, IHasReplayHandler, IFrameStableClock, IGameplayClock
|
public sealed class FrameStabilityContainer : Container, IHasReplayHandler, IFrameStableClock
|
||||||
{
|
{
|
||||||
public ReplayInputHandler? ReplayInputHandler { get; set; }
|
public ReplayInputHandler? ReplayInputHandler { get; set; }
|
||||||
|
|
||||||
|
@ -2,11 +2,11 @@
|
|||||||
// See the LICENCE file in the repository root for full licence text.
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
using osu.Framework.Bindables;
|
using osu.Framework.Bindables;
|
||||||
using osu.Framework.Timing;
|
using osu.Game.Screens.Play;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.UI
|
namespace osu.Game.Rulesets.UI
|
||||||
{
|
{
|
||||||
public interface IFrameStableClock : IFrameBasedClock
|
public interface IFrameStableClock : IGameplayClock
|
||||||
{
|
{
|
||||||
IBindable<bool> IsCatchingUp { get; }
|
IBindable<bool> IsCatchingUp { get; }
|
||||||
|
|
||||||
|
@ -20,11 +20,12 @@ using osu.Game.Input.Bindings;
|
|||||||
using osu.Game.Input.Handlers;
|
using osu.Game.Input.Handlers;
|
||||||
using osu.Game.Rulesets.Scoring;
|
using osu.Game.Rulesets.Scoring;
|
||||||
using osu.Game.Screens.Play;
|
using osu.Game.Screens.Play;
|
||||||
|
using osu.Game.Screens.Play.HUD.ClicksPerSecond;
|
||||||
using static osu.Game.Input.Handlers.ReplayInputHandler;
|
using static osu.Game.Input.Handlers.ReplayInputHandler;
|
||||||
|
|
||||||
namespace osu.Game.Rulesets.UI
|
namespace osu.Game.Rulesets.UI
|
||||||
{
|
{
|
||||||
public abstract class RulesetInputManager<T> : PassThroughInputManager, ICanAttachKeyCounter, IHasReplayHandler, IHasRecordingHandler
|
public abstract class RulesetInputManager<T> : PassThroughInputManager, ICanAttachHUDPieces, IHasReplayHandler, IHasRecordingHandler
|
||||||
where T : struct
|
where T : struct
|
||||||
{
|
{
|
||||||
public readonly KeyBindingContainer<T> KeyBindingContainer;
|
public readonly KeyBindingContainer<T> KeyBindingContainer;
|
||||||
@ -168,7 +169,7 @@ namespace osu.Game.Rulesets.UI
|
|||||||
.Select(action => new KeyCounterAction<T>(action)));
|
.Select(action => new KeyCounterAction<T>(action)));
|
||||||
}
|
}
|
||||||
|
|
||||||
public class ActionReceptor : KeyCounterDisplay.Receptor, IKeyBindingHandler<T>
|
private class ActionReceptor : KeyCounterDisplay.Receptor, IKeyBindingHandler<T>
|
||||||
{
|
{
|
||||||
public ActionReceptor(KeyCounterDisplay target)
|
public ActionReceptor(KeyCounterDisplay target)
|
||||||
: base(target)
|
: base(target)
|
||||||
@ -186,6 +187,37 @@ namespace osu.Game.Rulesets.UI
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
#region Keys per second Counter Attachment
|
||||||
|
|
||||||
|
public void Attach(ClicksPerSecondCalculator calculator)
|
||||||
|
{
|
||||||
|
var listener = new ActionListener(calculator);
|
||||||
|
|
||||||
|
KeyBindingContainer.Add(listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ActionListener : Component, IKeyBindingHandler<T>
|
||||||
|
{
|
||||||
|
private readonly ClicksPerSecondCalculator calculator;
|
||||||
|
|
||||||
|
public ActionListener(ClicksPerSecondCalculator calculator)
|
||||||
|
{
|
||||||
|
this.calculator = calculator;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool OnPressed(KeyBindingPressEvent<T> e)
|
||||||
|
{
|
||||||
|
calculator.AddInputTimestamp();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnReleased(KeyBindingReleaseEvent<T> e)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
protected virtual KeyBindingContainer<T> CreateKeyBindingContainer(RulesetInfo ruleset, int variant, SimultaneousBindingMode unique)
|
protected virtual KeyBindingContainer<T> CreateKeyBindingContainer(RulesetInfo ruleset, int variant, SimultaneousBindingMode unique)
|
||||||
=> new RulesetKeyBindingContainer(ruleset, variant, unique);
|
=> new RulesetKeyBindingContainer(ruleset, variant, unique);
|
||||||
|
|
||||||
@ -221,12 +253,13 @@ namespace osu.Game.Rulesets.UI
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Supports attaching a <see cref="KeyCounterDisplay"/>.
|
/// Supports attaching various HUD pieces.
|
||||||
/// Keys will be populated automatically and a receptor will be injected inside.
|
/// Keys will be populated automatically and a receptor will be injected inside.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public interface ICanAttachKeyCounter
|
public interface ICanAttachHUDPieces
|
||||||
{
|
{
|
||||||
void Attach(KeyCounterDisplay keyCounter);
|
void Attach(KeyCounterDisplay keyCounter);
|
||||||
|
void Attach(ClicksPerSecondCalculator calculator);
|
||||||
}
|
}
|
||||||
|
|
||||||
public class RulesetInputManagerInputState<T> : InputState
|
public class RulesetInputManagerInputState<T> : InputState
|
||||||
|
@ -0,0 +1,59 @@
|
|||||||
|
// 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.Collections.Generic;
|
||||||
|
using osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Game.Rulesets.UI;
|
||||||
|
|
||||||
|
namespace osu.Game.Screens.Play.HUD.ClicksPerSecond
|
||||||
|
{
|
||||||
|
public class ClicksPerSecondCalculator : Component
|
||||||
|
{
|
||||||
|
private readonly List<double> timestamps = new List<double>();
|
||||||
|
|
||||||
|
[Resolved]
|
||||||
|
private IGameplayClock gameplayClock { get; set; } = null!;
|
||||||
|
|
||||||
|
[Resolved(canBeNull: true)]
|
||||||
|
private DrawableRuleset? drawableRuleset { get; set; }
|
||||||
|
|
||||||
|
public int Value { get; private set; }
|
||||||
|
|
||||||
|
// Even though `FrameStabilityContainer` caches as a `GameplayClock`, we need to check it directly via `drawableRuleset`
|
||||||
|
// as this calculator is not contained within the `FrameStabilityContainer` and won't see the dependency.
|
||||||
|
private IGameplayClock clock => drawableRuleset?.FrameStableClock ?? gameplayClock;
|
||||||
|
|
||||||
|
public ClicksPerSecondCalculator()
|
||||||
|
{
|
||||||
|
RelativeSizeAxes = Axes.Both;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddInputTimestamp() => timestamps.Add(clock.CurrentTime);
|
||||||
|
|
||||||
|
protected override void Update()
|
||||||
|
{
|
||||||
|
base.Update();
|
||||||
|
|
||||||
|
double latestValidTime = clock.CurrentTime;
|
||||||
|
double earliestTimeValid = latestValidTime - 1000 * gameplayClock.GetTrueGameplayRate();
|
||||||
|
|
||||||
|
int count = 0;
|
||||||
|
|
||||||
|
for (int i = timestamps.Count - 1; i >= 0; i--)
|
||||||
|
{
|
||||||
|
// handle rewinding by removing future timestamps as we go
|
||||||
|
if (timestamps[i] > latestValidTime)
|
||||||
|
{
|
||||||
|
timestamps.RemoveAt(i);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (timestamps[i] >= earliestTimeValid)
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
Value = count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,102 @@
|
|||||||
|
// 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 osu.Framework.Allocation;
|
||||||
|
using osu.Framework.Graphics;
|
||||||
|
using osu.Framework.Graphics.Containers;
|
||||||
|
using osu.Framework.Graphics.Sprites;
|
||||||
|
using osu.Framework.Localisation;
|
||||||
|
using osu.Game.Graphics;
|
||||||
|
using osu.Game.Graphics.Sprites;
|
||||||
|
using osu.Game.Graphics.UserInterface;
|
||||||
|
using osu.Game.Skinning;
|
||||||
|
using osuTK;
|
||||||
|
|
||||||
|
namespace osu.Game.Screens.Play.HUD.ClicksPerSecond
|
||||||
|
{
|
||||||
|
public class ClicksPerSecondCounter : RollingCounter<int>, ISkinnableDrawable
|
||||||
|
{
|
||||||
|
[Resolved]
|
||||||
|
private ClicksPerSecondCalculator calculator { get; set; } = null!;
|
||||||
|
|
||||||
|
protected override double RollingDuration => 350;
|
||||||
|
|
||||||
|
public bool UsesFixedAnchor { get; set; }
|
||||||
|
|
||||||
|
public ClicksPerSecondCounter()
|
||||||
|
{
|
||||||
|
Current.Value = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
[BackgroundDependencyLoader]
|
||||||
|
private void load(OsuColour colours)
|
||||||
|
{
|
||||||
|
Colour = colours.BlueLighter;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Update()
|
||||||
|
{
|
||||||
|
base.Update();
|
||||||
|
|
||||||
|
Current.Value = calculator.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override IHasText CreateText() => new TextComponent();
|
||||||
|
|
||||||
|
private class TextComponent : CompositeDrawable, IHasText
|
||||||
|
{
|
||||||
|
public LocalisableString Text
|
||||||
|
{
|
||||||
|
get => text.Text;
|
||||||
|
set => text.Text = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly OsuSpriteText text;
|
||||||
|
|
||||||
|
public TextComponent()
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both;
|
||||||
|
|
||||||
|
InternalChild = new FillFlowContainer
|
||||||
|
{
|
||||||
|
AutoSizeAxes = Axes.Both,
|
||||||
|
Spacing = new Vector2(2),
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
text = new OsuSpriteText
|
||||||
|
{
|
||||||
|
Anchor = Anchor.BottomLeft,
|
||||||
|
Origin = Anchor.BottomLeft,
|
||||||
|
Font = OsuFont.Numeric.With(size: 16, fixedWidth: true)
|
||||||
|
},
|
||||||
|
new FillFlowContainer
|
||||||
|
{
|
||||||
|
Anchor = Anchor.BottomLeft,
|
||||||
|
Origin = Anchor.BottomLeft,
|
||||||
|
Direction = FillDirection.Vertical,
|
||||||
|
AutoSizeAxes = Axes.Both,
|
||||||
|
Children = new Drawable[]
|
||||||
|
{
|
||||||
|
new OsuSpriteText
|
||||||
|
{
|
||||||
|
Anchor = Anchor.TopLeft,
|
||||||
|
Origin = Anchor.TopLeft,
|
||||||
|
Font = OsuFont.Numeric.With(size: 6, fixedWidth: false),
|
||||||
|
Text = @"clicks",
|
||||||
|
},
|
||||||
|
new OsuSpriteText
|
||||||
|
{
|
||||||
|
Anchor = Anchor.TopLeft,
|
||||||
|
Origin = Anchor.TopLeft,
|
||||||
|
Font = OsuFont.Numeric.With(size: 6, fixedWidth: false),
|
||||||
|
Text = @"/sec",
|
||||||
|
Padding = new MarginPadding { Bottom = 3f }, // align baseline better
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -22,6 +22,7 @@ using osu.Game.Rulesets.Mods;
|
|||||||
using osu.Game.Rulesets.Scoring;
|
using osu.Game.Rulesets.Scoring;
|
||||||
using osu.Game.Rulesets.UI;
|
using osu.Game.Rulesets.UI;
|
||||||
using osu.Game.Screens.Play.HUD;
|
using osu.Game.Screens.Play.HUD;
|
||||||
|
using osu.Game.Screens.Play.HUD.ClicksPerSecond;
|
||||||
using osu.Game.Skinning;
|
using osu.Game.Skinning;
|
||||||
using osuTK;
|
using osuTK;
|
||||||
|
|
||||||
@ -49,6 +50,9 @@ namespace osu.Game.Screens.Play
|
|||||||
public readonly HoldForMenuButton HoldToQuit;
|
public readonly HoldForMenuButton HoldToQuit;
|
||||||
public readonly PlayerSettingsOverlay PlayerSettingsOverlay;
|
public readonly PlayerSettingsOverlay PlayerSettingsOverlay;
|
||||||
|
|
||||||
|
[Cached]
|
||||||
|
private readonly ClicksPerSecondCalculator clicksPerSecondCalculator;
|
||||||
|
|
||||||
public Bindable<bool> ShowHealthBar = new Bindable<bool>(true);
|
public Bindable<bool> ShowHealthBar = new Bindable<bool>(true);
|
||||||
|
|
||||||
private readonly DrawableRuleset drawableRuleset;
|
private readonly DrawableRuleset drawableRuleset;
|
||||||
@ -122,7 +126,8 @@ namespace osu.Game.Screens.Play
|
|||||||
KeyCounter = CreateKeyCounter(),
|
KeyCounter = CreateKeyCounter(),
|
||||||
HoldToQuit = CreateHoldForMenuButton(),
|
HoldToQuit = CreateHoldForMenuButton(),
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
clicksPerSecondCalculator = new ClicksPerSecondCalculator()
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -259,7 +264,11 @@ namespace osu.Game.Screens.Play
|
|||||||
|
|
||||||
protected virtual void BindDrawableRuleset(DrawableRuleset drawableRuleset)
|
protected virtual void BindDrawableRuleset(DrawableRuleset drawableRuleset)
|
||||||
{
|
{
|
||||||
(drawableRuleset as ICanAttachKeyCounter)?.Attach(KeyCounter);
|
if (drawableRuleset is ICanAttachHUDPieces attachTarget)
|
||||||
|
{
|
||||||
|
attachTarget.Attach(KeyCounter);
|
||||||
|
attachTarget.Attach(clicksPerSecondCalculator);
|
||||||
|
}
|
||||||
|
|
||||||
replayLoaded.BindTo(drawableRuleset.HasReplayLoaded);
|
replayLoaded.BindTo(drawableRuleset.HasReplayLoaded);
|
||||||
}
|
}
|
||||||
|
@ -1049,6 +1049,7 @@ namespace osu.Game.Screens.Play
|
|||||||
musicController.ResetTrackAdjustments();
|
musicController.ResetTrackAdjustments();
|
||||||
|
|
||||||
fadeOut();
|
fadeOut();
|
||||||
|
|
||||||
return base.OnExiting(e);
|
return base.OnExiting(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user