2019-01-24 17:43:03 +09: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.
2018-04-13 18:19:50 +09:00
2022-06-17 16:37:17 +09:00
#nullable disable
2019-03-06 20:30:14 +09:00
using System ;
2019-04-08 18:32:05 +09:00
using System.Collections.Generic ;
2023-02-16 19:20:06 +09:00
using JetBrains.Annotations ;
2017-03-11 14:27:18 +09:00
using osu.Framework.Allocation ;
2019-02-21 19:04:31 +09:00
using osu.Framework.Bindables ;
2021-05-11 12:42:32 +09:00
using osu.Framework.Extensions.EnumExtensions ;
2017-03-11 14:27:18 +09:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2023-07-28 15:48:21 +09:00
using osu.Framework.Graphics.Primitives ;
2020-10-30 14:19:40 +09:00
using osu.Framework.Input.Bindings ;
2021-09-16 18:26:12 +09:00
using osu.Framework.Input.Events ;
2017-03-11 14:27:18 +09:00
using osu.Game.Configuration ;
2020-10-30 14:19:40 +09:00
using osu.Game.Input.Bindings ;
2023-06-14 21:13:35 +02:00
using osu.Game.Localisation ;
2017-04-02 16:18:01 +03:00
using osu.Game.Overlays ;
using osu.Game.Overlays.Notifications ;
2023-06-14 21:13:35 +02:00
using osu.Game.Rulesets ;
2019-04-08 18:32:05 +09:00
using osu.Game.Rulesets.Mods ;
2022-02-03 14:50:40 +09:00
using osu.Game.Rulesets.Scoring ;
2017-05-05 13:00:05 +09:00
using osu.Game.Rulesets.UI ;
using osu.Game.Screens.Play.HUD ;
2022-08-14 20:12:11 +02:00
using osu.Game.Screens.Play.HUD.ClicksPerSecond ;
2023-01-17 09:55:45 +01:00
using osu.Game.Screens.Play.HUD.JudgementCounter ;
2021-04-30 12:42:32 +09:00
using osu.Game.Skinning ;
2018-11-20 16:51:59 +09:00
using osuTK ;
2018-04-13 18:19:50 +09:00
2017-05-05 13:00:05 +09:00
namespace osu.Game.Screens.Play
2017-03-11 14:27:18 +09:00
{
2020-10-14 17:21:56 +09:00
[Cached]
2021-05-10 22:43:48 +09:00
public partial class HUDOverlay : Container , IKeyBindingHandler < GlobalAction >
2017-03-11 14:27:18 +09:00
{
2021-04-14 14:25:16 +09:00
public const float FADE_DURATION = 300 ;
2020-10-15 16:56:05 +09:00
2021-04-14 14:25:16 +09:00
public const Easing FADE_EASING = Easing . OutQuint ;
2018-04-13 18:19:50 +09:00
2021-05-17 18:26:30 +09:00
/// <summary>
/// The total height of all the bottom of screen scoring elements.
/// </summary>
public float BottomScoringElementsHeight { get ; private set ; }
2022-10-30 21:38:45 +09:00
protected override bool ShouldBeConsideredForInput ( Drawable child )
{
// HUD uses AlwaysVisible on child components so they can be in an updated state for next display.
// Without blocking input, this would also allow them to be interacted with in such a state.
if ( ShowHud . Value )
return base . ShouldBeConsideredForInput ( child ) ;
// hold to quit button should always be interactive.
return child = = bottomRightElements ;
}
2022-10-10 15:57:41 +09:00
2017-05-05 13:00:05 +09:00
public readonly ModDisplay ModDisplay ;
2018-11-07 01:47:02 +09:00
public readonly HoldForMenuButton HoldToQuit ;
2018-01-13 22:25:09 +03:00
public readonly PlayerSettingsOverlay PlayerSettingsOverlay ;
2018-04-13 18:19:50 +09:00
2022-08-14 18:53:00 +02:00
[Cached]
2023-06-27 16:38:46 +09:00
private readonly ClicksPerSecondController clicksPerSecondController ;
2022-08-08 21:27:46 +02:00
2023-06-14 21:13:35 +02:00
[Cached]
2023-06-26 19:27:42 +02:00
public readonly InputCountController InputCountController ;
2023-06-14 21:13:35 +02:00
2022-12-11 22:08:48 +01:00
[Cached]
2023-06-26 19:30:04 +02:00
private readonly JudgementCountController judgementCountController ;
2022-12-11 22:08:48 +01:00
2021-12-09 21:15:00 -08:00
public Bindable < bool > ShowHealthBar = new Bindable < bool > ( true ) ;
2019-07-07 05:30:53 +09:00
2023-07-29 01:07:08 +02:00
[CanBeNull]
2019-05-09 18:06:11 +09:00
private readonly DrawableRuleset drawableRuleset ;
2023-07-29 01:07:08 +02:00
2019-05-09 18:06:11 +09:00
private readonly IReadOnlyList < Mod > mods ;
2019-12-12 15:05:29 +09:00
/// <summary>
/// Whether the elements that can optionally be hidden should be visible.
/// </summary>
public Bindable < bool > ShowHud { get ; } = new BindableBool ( ) ;
2020-07-22 12:41:06 +09:00
private Bindable < HUDVisibilityMode > configVisibilityMode ;
2023-07-28 14:39:41 -04:00
private Bindable < bool > configLeaderboardVisibility ;
2023-07-06 00:46:28 +09:00
private Bindable < bool > configSettingsOverlay ;
2019-12-12 15:05:29 +09:00
2018-01-17 17:37:14 +09:00
private readonly BindableBool replayLoaded = new BindableBool ( ) ;
2018-04-13 18:19:50 +09:00
2017-04-03 14:58:38 +03:00
private static bool hasShownNotificationOnce ;
2018-04-13 18:19:50 +09:00
2020-10-14 19:16:25 +09:00
private readonly FillFlowContainer bottomRightElements ;
2020-10-15 17:11:02 +09:00
private readonly FillFlowContainer topRightElements ;
2019-07-07 05:30:53 +09:00
2022-04-26 11:19:19 +09:00
internal readonly IBindable < bool > IsPlaying = new Bindable < bool > ( ) ;
2020-07-22 12:41:06 +09:00
2022-05-11 16:09:16 +09:00
public IBindable < bool > HoldingForHUD = > holdingForHUD ;
private readonly BindableBool holdingForHUD = new BindableBool ( ) ;
2020-10-30 14:19:40 +09:00
2023-02-15 18:31:55 +09:00
private readonly SkinComponentsContainer mainComponents ;
2021-05-11 12:42:32 +09:00
2024-01-18 15:44:27 +09:00
[CanBeNull]
private readonly SkinComponentsContainer rulesetComponents ;
2022-09-13 18:23:47 +09:00
/// <summary>
/// A flow which sits at the left side of the screen to house leaderboard (and related) components.
/// Will automatically be positioned to avoid colliding with top scoring elements.
/// </summary>
public readonly FillFlowContainer LeaderboardFlow ;
2022-09-13 18:12:49 +09:00
private readonly List < Drawable > hideTargets ;
2019-12-12 15:05:29 +09:00
2024-05-27 14:38:43 +09:00
/// <summary>
2024-05-27 08:26:00 +02:00
/// The container for skin components attached to <see cref="SkinComponentsContainerLookup.TargetArea.Playfield"/>
2024-05-27 14:38:43 +09:00
/// </summary>
internal readonly Drawable PlayfieldSkinLayer ;
2023-07-28 15:48:21 +09:00
2023-07-29 01:07:08 +02:00
public HUDOverlay ( [ CanBeNull ] DrawableRuleset drawableRuleset , IReadOnlyList < Mod > mods , bool alwaysShowLeaderboard = true )
2017-03-11 14:27:18 +09:00
{
2019-05-09 18:06:11 +09:00
this . drawableRuleset = drawableRuleset ;
this . mods = mods ;
2017-03-11 14:27:18 +09:00
RelativeSizeAxes = Axes . Both ;
2018-04-13 18:19:50 +09:00
2023-02-17 18:40:32 +09:00
Children = new [ ]
2017-03-11 14:27:18 +09:00
{
2021-05-07 17:27:34 +09:00
CreateFailingLayer ( ) ,
2022-12-14 23:00:34 +01:00
//Needs to be initialized before skinnable drawables.
2023-06-26 19:30:04 +02:00
judgementCountController = new JudgementCountController ( ) ,
2023-06-27 16:38:46 +09:00
clicksPerSecondController = new ClicksPerSecondController ( ) ,
2023-06-27 16:37:22 +09:00
InputCountController = new InputCountController ( ) ,
2023-02-16 19:20:06 +09:00
mainComponents = new HUDComponentsContainer { AlwaysPresent = true , } ,
2024-01-18 15:44:27 +09:00
drawableRuleset ! = null
? ( rulesetComponents = new HUDComponentsContainer ( drawableRuleset . Ruleset . RulesetInfo ) { AlwaysPresent = true , } )
2023-02-17 18:40:32 +09:00
: Empty ( ) ,
2024-05-27 14:38:43 +09:00
PlayfieldSkinLayer = drawableRuleset ! = null
2023-07-28 15:48:21 +09:00
? new SkinComponentsContainer ( new SkinComponentsContainerLookup ( SkinComponentsContainerLookup . TargetArea . Playfield , drawableRuleset . Ruleset . RulesetInfo ) ) { AlwaysPresent = true , }
: Empty ( ) ,
2020-10-15 17:11:02 +09:00
topRightElements = new FillFlowContainer
{
Anchor = Anchor . TopRight ,
Origin = Anchor . TopRight ,
2022-06-15 01:04:43 +09:00
AlwaysPresent = true ,
2020-10-15 17:11:02 +09:00
Margin = new MarginPadding ( 10 ) ,
Spacing = new Vector2 ( 10 ) ,
AutoSizeAxes = Axes . Both ,
Direction = FillDirection . Vertical ,
Children = new Drawable [ ]
{
2019-01-17 16:00:11 +09:00
ModDisplay = CreateModsContainer ( ) ,
2019-12-12 14:19:04 +09:00
PlayerSettingsOverlay = CreatePlayerSettingsOverlay ( ) ,
2019-01-17 16:00:11 +09:00
}
} ,
2020-10-14 18:51:53 +09:00
bottomRightElements = new FillFlowContainer
2017-04-03 11:41:17 +03:00
{
2019-01-17 16:00:11 +09:00
Anchor = Anchor . BottomRight ,
Origin = Anchor . BottomRight ,
2020-10-15 17:11:02 +09:00
Margin = new MarginPadding ( 10 ) ,
Spacing = new Vector2 ( 10 ) ,
2019-01-17 16:00:11 +09:00
AutoSizeAxes = Axes . Both ,
2020-10-15 16:56:05 +09:00
LayoutDuration = FADE_DURATION / 2 ,
LayoutEasing = FADE_EASING ,
2019-01-17 16:00:11 +09:00
Direction = FillDirection . Vertical ,
Children = new Drawable [ ]
2018-05-22 16:45:42 +09:00
{
2019-01-17 16:00:11 +09:00
HoldToQuit = CreateHoldForMenuButton ( ) ,
2018-05-22 16:45:42 +09:00
}
2022-08-08 21:27:46 +02:00
} ,
2022-09-13 18:23:47 +09:00
LeaderboardFlow = new FillFlowContainer
{
AutoSizeAxes = Axes . Both ,
Direction = FillDirection . Vertical ,
2022-09-13 18:42:57 +09:00
Padding = new MarginPadding ( 44 ) , // enough margin to avoid the hit error display
2022-09-13 18:23:47 +09:00
Spacing = new Vector2 ( 5 )
} ,
2019-01-17 16:00:11 +09:00
} ;
2022-09-13 18:12:49 +09:00
2024-05-02 15:36:40 +08:00
hideTargets = new List < Drawable > { mainComponents , topRightElements } ;
2024-01-22 13:45:55 +01:00
if ( rulesetComponents ! = null )
hideTargets . Add ( rulesetComponents ) ;
2022-09-13 18:23:47 +09:00
2022-09-13 19:57:40 +09:00
if ( ! alwaysShowLeaderboard )
2022-09-13 18:23:47 +09:00
hideTargets . Add ( LeaderboardFlow ) ;
2019-05-09 18:06:11 +09:00
}
2018-04-13 18:19:50 +09:00
2019-05-09 18:06:11 +09:00
[BackgroundDependencyLoader(true)]
2022-08-11 11:58:30 +02:00
private void load ( OsuConfigManager config , INotificationOverlay notificationOverlay )
2019-05-09 18:06:11 +09:00
{
2019-12-12 16:09:42 +09:00
if ( drawableRuleset ! = null )
{
BindDrawableRuleset ( drawableRuleset ) ;
}
2018-04-13 18:19:50 +09:00
2019-04-08 18:32:05 +09:00
ModDisplay . Current . Value = mods ;
2018-04-13 18:19:50 +09:00
2020-07-22 12:41:06 +09:00
configVisibilityMode = config . GetBindable < HUDVisibilityMode > ( OsuSetting . HUDVisibilityMode ) ;
2023-07-28 14:39:41 -04:00
configLeaderboardVisibility = config . GetBindable < bool > ( OsuSetting . GameplayLeaderboard ) ;
2023-07-06 00:46:28 +09:00
configSettingsOverlay = config . GetBindable < bool > ( OsuSetting . ReplaySettingsOverlay ) ;
2019-12-12 15:05:29 +09:00
2020-07-22 12:41:06 +09:00
if ( configVisibilityMode . Value = = HUDVisibilityMode . Never & & ! hasShownNotificationOnce )
2019-12-12 15:05:29 +09:00
{
hasShownNotificationOnce = true ;
notificationOverlay ? . Post ( new SimpleNotification
{
2023-01-14 02:30:10 +03:00
Text = NotificationsStrings . ScoreOverlayDisabled ( config . LookupKeyBindings ( GlobalAction . ToggleInGameInterface ) )
2019-12-12 15:05:29 +09:00
} ) ;
}
// start all elements hidden
hideTargets . ForEach ( d = > d . Hide ( ) ) ;
}
2022-05-11 16:09:16 +09:00
public override void Hide ( ) = >
throw new InvalidOperationException ( $"{nameof(HUDOverlay)} should not be hidden as it will remove the ability of a user to quit. Use {nameof(ShowHud)} instead." ) ;
2019-12-12 15:05:29 +09:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2020-10-15 16:56:05 +09:00
ShowHud . BindValueChanged ( visible = > hideTargets . ForEach ( d = > d . FadeTo ( visible . NewValue ? 1 : 0 , FADE_DURATION , FADE_EASING ) ) ) ;
2018-04-13 18:19:50 +09:00
2022-05-11 16:09:16 +09:00
holdingForHUD . BindValueChanged ( _ = > updateVisibility ( ) ) ;
2022-04-26 11:19:19 +09:00
IsPlaying . BindValueChanged ( _ = > updateVisibility ( ) ) ;
2023-07-06 00:46:28 +09:00
configVisibilityMode . BindValueChanged ( _ = > updateVisibility ( ) ) ;
configSettingsOverlay . BindValueChanged ( _ = > updateVisibility ( ) ) ;
2018-04-13 18:19:50 +09:00
2023-07-06 00:46:28 +09:00
replayLoaded . BindValueChanged ( e = >
{
if ( e . NewValue )
{
ModDisplay . FadeIn ( 200 ) ;
InputCountController . Margin = new MarginPadding ( 10 ) { Bottom = 30 } ;
}
else
{
ModDisplay . Delay ( 2000 ) . FadeOut ( 200 ) ;
InputCountController . Margin = new MarginPadding ( 10 ) ;
}
updateVisibility ( ) ;
} , true ) ;
2018-01-17 21:52:57 +09:00
}
2018-04-13 18:19:50 +09:00
2020-10-14 18:51:53 +09:00
protected override void Update ( )
{
base . Update ( ) ;
2020-10-15 17:11:02 +09:00
2023-07-29 01:07:08 +02:00
if ( drawableRuleset ! = null )
{
Quad playfieldScreenSpaceDrawQuad = drawableRuleset . Playfield . SkinnableComponentScreenSpaceDrawQuad ;
2023-07-28 15:48:21 +09:00
2024-05-27 14:38:43 +09:00
PlayfieldSkinLayer . Position = ToLocalSpace ( playfieldScreenSpaceDrawQuad . TopLeft ) ;
PlayfieldSkinLayer . Width = ( ToLocalSpace ( playfieldScreenSpaceDrawQuad . TopRight ) - ToLocalSpace ( playfieldScreenSpaceDrawQuad . TopLeft ) ) . Length ;
PlayfieldSkinLayer . Height = ( ToLocalSpace ( playfieldScreenSpaceDrawQuad . BottomLeft ) - ToLocalSpace ( playfieldScreenSpaceDrawQuad . TopLeft ) ) . Length ;
PlayfieldSkinLayer . Rotation = drawableRuleset . Playfield . Rotation ;
2023-07-29 01:07:08 +02:00
}
2023-07-28 15:48:21 +09:00
2022-09-13 18:42:57 +09:00
float? lowestTopScreenSpaceLeft = null ;
float? lowestTopScreenSpaceRight = null ;
2021-05-17 18:26:30 +09:00
Vector2 ? highestBottomScreenSpace = null ;
2021-05-11 12:42:32 +09:00
2024-02-26 09:04:39 +08:00
processDrawables ( mainComponents ) ;
2024-01-18 15:44:27 +09:00
if ( rulesetComponents ! = null )
2024-02-26 09:04:39 +08:00
processDrawables ( rulesetComponents ) ;
2024-01-18 15:44:27 +09:00
if ( lowestTopScreenSpaceRight . HasValue )
topRightElements . Y = MathHelper . Clamp ( ToLocalSpace ( new Vector2 ( 0 , lowestTopScreenSpaceRight . Value ) ) . Y , 0 , DrawHeight - topRightElements . DrawHeight ) ;
else
topRightElements . Y = 0 ;
if ( lowestTopScreenSpaceLeft . HasValue )
LeaderboardFlow . Y = MathHelper . Clamp ( ToLocalSpace ( new Vector2 ( 0 , lowestTopScreenSpaceLeft . Value ) ) . Y , 0 , DrawHeight - LeaderboardFlow . DrawHeight ) ;
else
LeaderboardFlow . Y = 0 ;
if ( highestBottomScreenSpace . HasValue )
bottomRightElements . Y = BottomScoringElementsHeight = - MathHelper . Clamp ( DrawHeight - ToLocalSpace ( highestBottomScreenSpace . Value ) . Y , 0 , DrawHeight - bottomRightElements . DrawHeight ) ;
else
bottomRightElements . Y = 0 ;
2024-02-26 09:04:39 +08:00
void processDrawables ( SkinComponentsContainer components )
{
// Avoid using foreach due to missing GetEnumerator implementation.
// See https://github.com/ppy/osu-framework/blob/e10051e6643731e393b09de40a3a3d209a545031/osu.Framework/Bindables/IBindableList.cs#L41-L44.
for ( int i = 0 ; i < components . Components . Count ; i + + )
processDrawable ( components . Components [ i ] ) ;
}
2024-01-26 12:55:36 +09:00
void processDrawable ( ISerialisableDrawable element )
2021-05-11 12:42:32 +09:00
{
2024-01-26 12:55:36 +09:00
// Cast can be removed when IDrawable interface includes Anchor / RelativeSizeAxes.
Drawable drawable = ( Drawable ) element ;
2022-09-20 00:08:06 +09:00
// for now align some top components with the bottom-edge of the lowest top-anchored hud element.
2024-01-26 12:55:36 +09:00
if ( drawable . Anchor . HasFlagFast ( Anchor . y0 ) )
2021-05-17 18:26:30 +09:00
{
// health bars are excluded for the sake of hacky legacy skins which extend the health bar to take up the full screen area.
if ( element is LegacyHealthDisplay )
2024-01-18 15:44:27 +09:00
return ;
2020-10-15 18:30:44 +09:00
2024-01-26 12:55:36 +09:00
float bottom = drawable . ScreenSpaceDrawQuad . BottomRight . Y ;
2022-09-13 18:42:57 +09:00
2024-01-26 12:55:36 +09:00
bool isRelativeX = drawable . RelativeSizeAxes = = Axes . X ;
2022-09-13 18:42:57 +09:00
2024-01-26 12:55:36 +09:00
if ( drawable . Anchor . HasFlagFast ( Anchor . TopRight ) | | isRelativeX )
2022-09-13 18:42:57 +09:00
{
if ( lowestTopScreenSpaceRight = = null | | bottom > lowestTopScreenSpaceRight . Value )
lowestTopScreenSpaceRight = bottom ;
}
2024-01-26 12:55:36 +09:00
if ( drawable . Anchor . HasFlagFast ( Anchor . TopLeft ) | | isRelativeX )
2022-09-13 18:42:57 +09:00
{
if ( lowestTopScreenSpaceLeft = = null | | bottom > lowestTopScreenSpaceLeft . Value )
lowestTopScreenSpaceLeft = bottom ;
}
2021-05-17 18:26:30 +09:00
}
2021-05-23 16:22:51 +03:00
// and align bottom-right components with the top-edge of the highest bottom-anchored hud element.
2024-01-26 12:55:36 +09:00
else if ( drawable . Anchor . HasFlagFast ( Anchor . BottomRight ) | | ( drawable . Anchor . HasFlagFast ( Anchor . y2 ) & & drawable . RelativeSizeAxes = = Axes . X ) )
2021-05-17 18:26:30 +09:00
{
var topLeft = element . ScreenSpaceDrawQuad . TopLeft ;
if ( highestBottomScreenSpace = = null | | topLeft . Y < highestBottomScreenSpace . Value . Y )
highestBottomScreenSpace = topLeft ;
}
2021-05-11 12:42:32 +09:00
}
2020-10-14 18:51:53 +09:00
}
2020-07-22 12:41:06 +09:00
private void updateVisibility ( )
{
if ( ShowHud . Disabled )
return ;
2022-05-11 16:09:16 +09:00
if ( holdingForHUD . Value )
2020-10-30 14:19:40 +09:00
{
ShowHud . Value = true ;
return ;
}
2023-07-06 00:46:28 +09:00
if ( configSettingsOverlay . Value & & replayLoaded . Value )
PlayerSettingsOverlay . Show ( ) ;
else
PlayerSettingsOverlay . Hide ( ) ;
2020-07-22 12:41:06 +09:00
switch ( configVisibilityMode . Value )
{
case HUDVisibilityMode . Never :
ShowHud . Value = false ;
break ;
2020-10-20 14:19:04 +09:00
case HUDVisibilityMode . HideDuringGameplay :
// always show during replay as we want the seek bar to be visible.
2022-04-26 11:19:19 +09:00
ShowHud . Value = replayLoaded . Value | | ! IsPlaying . Value ;
2020-10-20 14:19:04 +09:00
break ;
2020-07-22 12:41:06 +09:00
case HUDVisibilityMode . Always :
ShowHud . Value = true ;
break ;
}
}
2019-03-19 23:44:15 +09:00
protected virtual void BindDrawableRuleset ( DrawableRuleset drawableRuleset )
2017-03-11 14:27:18 +09:00
{
2023-06-25 15:04:39 +02:00
if ( drawableRuleset is ICanAttachHUDPieces attachTarget )
{
2023-06-26 19:27:42 +02:00
attachTarget . Attach ( InputCountController ) ;
2023-06-27 16:38:46 +09:00
attachTarget . Attach ( clicksPerSecondController ) ;
2023-06-25 15:04:39 +02:00
}
2018-04-13 18:19:50 +09:00
2019-03-19 23:44:15 +09:00
replayLoaded . BindTo ( drawableRuleset . HasReplayLoaded ) ;
2017-03-11 14:27:18 +09:00
}
2018-04-13 18:19:50 +09:00
2021-05-06 15:16:16 +09:00
protected FailingLayer CreateFailingLayer ( ) = > new FailingLayer
2020-06-26 19:22:30 +02:00
{
2021-12-09 21:15:00 -08:00
ShowHealth = { BindTarget = ShowHealthBar }
2020-06-26 19:22:30 +02:00
} ;
2020-04-09 14:31:25 +09:00
2021-05-06 15:16:16 +09:00
protected HoldForMenuButton CreateHoldForMenuButton ( ) = > new HoldForMenuButton
2018-04-21 18:24:31 +03:00
{
Anchor = Anchor . BottomRight ,
Origin = Anchor . BottomRight ,
} ;
2021-05-06 15:16:16 +09:00
protected ModDisplay CreateModsContainer ( ) = > new ModDisplay
2017-06-05 17:15:04 +09:00
{
Anchor = Anchor . TopRight ,
Origin = Anchor . TopRight ,
} ;
2018-04-13 18:19:50 +09:00
2021-05-06 15:16:16 +09:00
protected PlayerSettingsOverlay CreatePlayerSettingsOverlay ( ) = > new PlayerSettingsOverlay ( ) ;
2018-04-13 18:19:50 +09:00
2021-09-16 18:26:12 +09:00
public bool OnPressed ( KeyBindingPressEvent < GlobalAction > e )
2020-10-30 14:19:40 +09:00
{
2021-11-18 12:35:47 +09:00
if ( e . Repeat )
return false ;
2021-09-16 18:26:12 +09:00
switch ( e . Action )
2020-10-30 14:19:40 +09:00
{
2023-07-06 01:00:41 +09:00
case GlobalAction . ToggleReplaySettings :
configSettingsOverlay . Value = ! configSettingsOverlay . Value ;
return true ;
2020-10-30 14:19:40 +09:00
case GlobalAction . HoldForHUD :
2022-05-11 16:09:16 +09:00
holdingForHUD . Value = true ;
2020-10-30 14:19:40 +09:00
return true ;
2020-11-29 20:59:02 -05:00
case GlobalAction . ToggleInGameInterface :
switch ( configVisibilityMode . Value )
{
case HUDVisibilityMode . Never :
configVisibilityMode . Value = HUDVisibilityMode . HideDuringGameplay ;
break ;
case HUDVisibilityMode . HideDuringGameplay :
configVisibilityMode . Value = HUDVisibilityMode . Always ;
break ;
case HUDVisibilityMode . Always :
configVisibilityMode . Value = HUDVisibilityMode . Never ;
break ;
}
2020-12-01 14:00:54 +09:00
2020-11-29 20:59:02 -05:00
return true ;
2023-07-28 14:39:41 -04:00
case GlobalAction . ToggleInGameLeaderboard :
configLeaderboardVisibility . Value = ! configLeaderboardVisibility . Value ;
return true ;
2020-10-30 14:19:40 +09:00
}
return false ;
}
2021-09-16 18:26:12 +09:00
public void OnReleased ( KeyBindingReleaseEvent < GlobalAction > e )
2020-10-30 14:19:40 +09:00
{
2021-09-16 18:26:12 +09:00
switch ( e . Action )
2020-10-30 14:19:40 +09:00
{
case GlobalAction . HoldForHUD :
2022-05-11 16:09:16 +09:00
holdingForHUD . Value = false ;
2020-10-30 14:19:40 +09:00
break ;
}
}
2022-02-03 14:50:40 +09:00
2023-02-16 19:20:06 +09:00
private partial class HUDComponentsContainer : SkinComponentsContainer
2022-02-03 14:50:40 +09:00
{
private Bindable < ScoringMode > scoringMode ;
[Resolved]
private OsuConfigManager config { get ; set ; }
2023-02-17 15:00:57 +09:00
public HUDComponentsContainer ( [ CanBeNull ] RulesetInfo ruleset = null )
2023-02-16 19:20:06 +09:00
: base ( new SkinComponentsContainerLookup ( SkinComponentsContainerLookup . TargetArea . MainHUDComponents , ruleset ) )
2022-02-03 14:50:40 +09:00
{
RelativeSizeAxes = Axes . Both ;
}
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
// When the scoring mode changes, relative positions of elements may change (see DefaultSkin.GetDrawableComponent).
// This is a best effort implementation for cases where users haven't customised layouts.
scoringMode = config . GetBindable < ScoringMode > ( OsuSetting . ScoreDisplayMode ) ;
2022-06-24 21:25:23 +09:00
scoringMode . BindValueChanged ( _ = > Reload ( ) ) ;
2022-02-03 14:50:40 +09:00
}
}
2017-03-11 14:27:18 +09:00
}
}