From 9ca672d689ecf08c5f7c8a08d5e46ada684cac36 Mon Sep 17 00:00:00 2001 From: Salman Ahmed Date: Mon, 23 Oct 2023 01:58:41 +0300 Subject: [PATCH] Add "Argon" total score counter --- .../TestSceneSkinnableScoreCounter.cs | 1 + .../Screens/Play/HUD/ArgonScoreCounter.cs | 124 ++++++++++++++++++ 2 files changed, 125 insertions(+) create mode 100644 osu.Game/Screens/Play/HUD/ArgonScoreCounter.cs diff --git a/osu.Game.Tests/Visual/Gameplay/TestSceneSkinnableScoreCounter.cs b/osu.Game.Tests/Visual/Gameplay/TestSceneSkinnableScoreCounter.cs index 7079b93d3e..186680a79c 100644 --- a/osu.Game.Tests/Visual/Gameplay/TestSceneSkinnableScoreCounter.cs +++ b/osu.Game.Tests/Visual/Gameplay/TestSceneSkinnableScoreCounter.cs @@ -17,6 +17,7 @@ namespace osu.Game.Tests.Visual.Gameplay [Cached(typeof(ScoreProcessor))] private ScoreProcessor scoreProcessor = TestGameplayState.Create(new OsuRuleset()).ScoreProcessor; + protected override Drawable CreateArgonImplementation() => new ArgonScoreCounter(); protected override Drawable CreateDefaultImplementation() => new DefaultScoreCounter(); protected override Drawable CreateLegacyImplementation() => new LegacyScoreCounter(); diff --git a/osu.Game/Screens/Play/HUD/ArgonScoreCounter.cs b/osu.Game/Screens/Play/HUD/ArgonScoreCounter.cs new file mode 100644 index 0000000000..cfda34578b --- /dev/null +++ b/osu.Game/Screens/Play/HUD/ArgonScoreCounter.cs @@ -0,0 +1,124 @@ +// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. +// See the LICENCE file in the repository root for full licence text. + +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.Sprites; +using osu.Framework.Localisation; +using osu.Framework.Text; +using osu.Game.Configuration; +using osu.Game.Graphics.Sprites; +using osu.Game.Skinning; +using osuTK; + +namespace osu.Game.Screens.Play.HUD +{ + public partial class ArgonScoreCounter : GameplayScoreCounter, ISerialisableDrawable + { + [SettingSource("Wireframe opacity", "Controls the opacity of the wire frames behind the digits.")] + public BindableFloat WireframeOpactiy { get; } = new BindableFloat(0.4f) + { + Precision = 0.01f, + MinValue = 0, + MaxValue = 1, + }; + + public bool UsesFixedAnchor { get; set; } + + protected override IHasText CreateText() => new ArgonScoreTextComponent + { + WireframeOpactiy = { BindTarget = WireframeOpactiy }, + }; + + private partial class ArgonScoreTextComponent : CompositeDrawable, IHasText + { + private readonly ArgonScoreSpriteText wireframesPart; + private readonly ArgonScoreSpriteText textPart; + + public IBindable WireframeOpactiy { get; } = new BindableFloat(); + + public LocalisableString Text + { + get => textPart.Text; + set + { + wireframesPart.Text = value; + textPart.Text = value; + } + } + + public ArgonScoreTextComponent() + { + AutoSizeAxes = Axes.Both; + + InternalChildren = new[] + { + wireframesPart = new ArgonScoreSpriteText(@"wireframes"), + textPart = new ArgonScoreSpriteText(), + }; + } + + protected override void LoadComplete() + { + base.LoadComplete(); + WireframeOpactiy.BindValueChanged(v => wireframesPart.Alpha = v.NewValue, true); + } + + private partial class ArgonScoreSpriteText : OsuSpriteText + { + private readonly string? glyphLookupOverride; + + private GlyphStore glyphStore = null!; + + protected override char FixedWidthReferenceCharacter => '5'; + + public ArgonScoreSpriteText(string? glyphLookupOverride = null) + { + this.glyphLookupOverride = glyphLookupOverride; + + Shadow = false; + UseFullGlyphHeight = false; + } + + [BackgroundDependencyLoader] + private void load(ISkinSource skin) + { + Font = new FontUsage(@"argon-score", 1, fixedWidth: true); + Spacing = new Vector2(-2, 0); + + glyphStore = new GlyphStore(skin, glyphLookupOverride); + } + + protected override TextBuilder CreateTextBuilder(ITexturedGlyphLookupStore store) => base.CreateTextBuilder(glyphStore); + + private class GlyphStore : ITexturedGlyphLookupStore + { + private readonly ISkin skin; + private readonly string? glyphLookupOverride; + + public GlyphStore(ISkin skin, string? glyphLookupOverride) + { + this.skin = skin; + this.glyphLookupOverride = glyphLookupOverride; + } + + public ITexturedCharacterGlyph? Get(string fontName, char character) + { + string lookup = glyphLookupOverride ?? character.ToString(); + var texture = skin.GetTexture($"{fontName}-{lookup}"); + + if (texture == null) + return null; + + return new TexturedCharacterGlyph(new CharacterGlyph(character, 0, 0, texture.Width, texture.Height, null), texture, 0.25f); + } + + public Task GetAsync(string fontName, char character) => Task.Run(() => Get(fontName, character)); + } + } + } + } +}