2019-01-24 08:43:03 +00: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 09:19:50 +00:00
2019-03-06 11:30:14 +00:00
using System ;
2019-04-08 09:32:05 +00:00
using System.Collections.Generic ;
2021-05-11 08:48:08 +00:00
using System.Linq ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Allocation ;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables ;
2021-05-11 03:42:32 +00:00
using osu.Framework.Extensions.EnumExtensions ;
2019-12-12 06:05:29 +00:00
using osu.Framework.Extensions.IEnumerableExtensions ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2020-10-30 05:19:40 +00:00
using osu.Framework.Input.Bindings ;
2018-04-13 09:19:50 +00:00
using osu.Game.Configuration ;
2020-10-30 05:19:40 +00:00
using osu.Game.Input.Bindings ;
2018-04-13 09:19:50 +00:00
using osu.Game.Overlays ;
using osu.Game.Overlays.Notifications ;
2019-04-08 09:32:05 +00:00
using osu.Game.Rulesets.Mods ;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.UI ;
using osu.Game.Screens.Play.HUD ;
2021-04-30 03:42:32 +00:00
using osu.Game.Skinning ;
2018-11-20 07:51:59 +00:00
using osuTK ;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Screens.Play
{
2020-10-14 08:21:56 +00:00
[Cached]
2021-05-10 13:43:48 +00:00
public class HUDOverlay : Container , IKeyBindingHandler < GlobalAction >
2018-04-13 09:19:50 +00:00
{
2021-04-14 05:25:16 +00:00
public const float FADE_DURATION = 300 ;
2020-10-15 07:56:05 +00:00
2021-04-14 05:25:16 +00:00
public const Easing FADE_EASING = Easing . OutQuint ;
2018-04-13 09:19:50 +00:00
2020-12-22 10:09:59 +00:00
/// <summary>
/// The total height of all the top of screen scoring elements.
/// </summary>
public float TopScoringElementsHeight { get ; private set ; }
2019-03-26 02:28:43 +00:00
public readonly KeyCounterDisplay KeyCounter ;
2018-04-13 09:19:50 +00:00
public readonly SongProgress Progress ;
public readonly ModDisplay ModDisplay ;
2018-11-06 16:47:02 +00:00
public readonly HoldForMenuButton HoldToQuit ;
2018-04-13 09:19:50 +00:00
public readonly PlayerSettingsOverlay PlayerSettingsOverlay ;
2019-07-06 20:30:53 +00:00
public Bindable < bool > ShowHealthbar = new Bindable < bool > ( true ) ;
2019-05-09 09:06:11 +00:00
private readonly DrawableRuleset drawableRuleset ;
private readonly IReadOnlyList < Mod > mods ;
2019-12-12 06:05:29 +00:00
/// <summary>
/// Whether the elements that can optionally be hidden should be visible.
/// </summary>
public Bindable < bool > ShowHud { get ; } = new BindableBool ( ) ;
2020-07-22 03:41:06 +00:00
private Bindable < HUDVisibilityMode > configVisibilityMode ;
2019-12-12 06:05:29 +00:00
2019-01-17 07:00:11 +00:00
private readonly Container visibilityContainer ;
2019-12-12 06:05:29 +00:00
2018-04-13 09:19:50 +00:00
private readonly BindableBool replayLoaded = new BindableBool ( ) ;
private static bool hasShownNotificationOnce ;
2019-03-06 11:30:14 +00:00
public Action < double > RequestSeek ;
2020-10-14 10:16:25 +00:00
private readonly FillFlowContainer bottomRightElements ;
2020-10-15 08:11:02 +00:00
private readonly FillFlowContainer topRightElements ;
2019-07-06 20:30:53 +00:00
2020-07-22 03:41:06 +00:00
internal readonly IBindable < bool > IsBreakTime = new Bindable < bool > ( ) ;
2020-10-30 05:19:40 +00:00
private bool holdingForHUD ;
2021-05-11 03:42:32 +00:00
private readonly SkinnableElementTargetContainer mainComponents ;
2021-04-14 05:16:18 +00:00
private IEnumerable < Drawable > hideTargets = > new Drawable [ ] { visibilityContainer , KeyCounter , topRightElements } ;
2019-12-12 06:05:29 +00:00
2021-05-07 08:27:34 +00:00
public HUDOverlay ( DrawableRuleset drawableRuleset , IReadOnlyList < Mod > mods )
2018-04-13 09:19:50 +00:00
{
2019-05-09 09:06:11 +00:00
this . drawableRuleset = drawableRuleset ;
this . mods = mods ;
2018-04-13 09:19:50 +00:00
RelativeSizeAxes = Axes . Both ;
2019-01-17 07:00:11 +00:00
Children = new Drawable [ ]
2018-04-13 09:19:50 +00:00
{
2021-05-07 08:27:34 +00:00
CreateFailingLayer ( ) ,
2019-01-23 05:51:13 +00:00
visibilityContainer = new Container
{
2019-01-17 07:00:11 +00:00
RelativeSizeAxes = Axes . Both ,
2020-10-14 09:15:29 +00:00
Child = new GridContainer
2019-01-23 05:51:13 +00:00
{
2020-10-14 09:15:29 +00:00
RelativeSizeAxes = Axes . Both ,
Content = new [ ]
2019-01-23 05:51:13 +00:00
{
2020-10-14 09:15:29 +00:00
new Drawable [ ]
2019-01-23 05:51:13 +00:00
{
2021-05-07 10:13:38 +00:00
new Container
2020-10-14 09:15:29 +00:00
{
RelativeSizeAxes = Axes . Both ,
Children = new Drawable [ ]
{
2021-05-11 03:42:32 +00:00
mainComponents = new SkinnableElementTargetContainer ( SkinnableTarget . MainHUDComponents )
2021-05-07 10:13:38 +00:00
{
RelativeSizeAxes = Axes . Both ,
} ,
new Container
{
RelativeSizeAxes = Axes . Both ,
Children = new Drawable [ ]
{
// still need to be migrated; a bit more involved.
new HitErrorDisplay ( this . drawableRuleset ? . FirstAvailableHitWindows ) ,
}
} ,
2020-10-14 09:15:29 +00:00
}
} ,
2019-01-21 08:03:06 +00:00
} ,
2020-10-14 09:15:29 +00:00
new Drawable [ ]
{
Progress = CreateProgress ( ) ,
}
2019-01-21 08:03:06 +00:00
} ,
2020-10-14 09:15:29 +00:00
RowDimensions = new [ ]
{
2020-10-14 09:51:53 +00:00
new Dimension ( ) ,
2020-10-14 09:15:29 +00:00
new Dimension ( GridSizeMode . AutoSize )
}
} ,
2019-01-17 07:00:11 +00:00
} ,
2020-10-15 08:11:02 +00:00
topRightElements = new FillFlowContainer
{
Anchor = Anchor . TopRight ,
Origin = Anchor . TopRight ,
Margin = new MarginPadding ( 10 ) ,
Spacing = new Vector2 ( 10 ) ,
AutoSizeAxes = Axes . Both ,
Direction = FillDirection . Vertical ,
Children = new Drawable [ ]
{
2019-01-17 07:00:11 +00:00
ModDisplay = CreateModsContainer ( ) ,
2019-12-12 05:19:04 +00:00
PlayerSettingsOverlay = CreatePlayerSettingsOverlay ( ) ,
2019-01-17 07:00:11 +00:00
}
} ,
2020-10-14 09:51:53 +00:00
bottomRightElements = new FillFlowContainer
2018-04-13 09:19:50 +00:00
{
2019-01-17 07:00:11 +00:00
Anchor = Anchor . BottomRight ,
Origin = Anchor . BottomRight ,
2020-10-15 08:11:02 +00:00
Margin = new MarginPadding ( 10 ) ,
Spacing = new Vector2 ( 10 ) ,
2019-01-17 07:00:11 +00:00
AutoSizeAxes = Axes . Both ,
2020-10-15 07:56:05 +00:00
LayoutDuration = FADE_DURATION / 2 ,
LayoutEasing = FADE_EASING ,
2019-01-17 07:00:11 +00:00
Direction = FillDirection . Vertical ,
Children = new Drawable [ ]
2018-05-22 07:45:42 +00:00
{
2019-03-05 04:26:54 +00:00
KeyCounter = CreateKeyCounter ( ) ,
2019-01-17 07:00:11 +00:00
HoldToQuit = CreateHoldForMenuButton ( ) ,
2018-05-22 07:45:42 +00:00
}
2018-04-13 09:19:50 +00:00
}
2019-01-17 07:00:11 +00:00
} ;
2019-05-09 09:06:11 +00:00
}
2018-04-13 09:19:50 +00:00
2019-05-09 09:06:11 +00:00
[BackgroundDependencyLoader(true)]
private void load ( OsuConfigManager config , NotificationOverlay notificationOverlay )
{
2019-12-12 07:09:42 +00:00
if ( drawableRuleset ! = null )
{
BindDrawableRuleset ( drawableRuleset ) ;
Progress . Objects = drawableRuleset . Objects ;
Progress . RequestSeek = time = > RequestSeek ( time ) ;
Progress . ReferenceClock = drawableRuleset . FrameStableClock ;
}
2018-04-13 09:19:50 +00:00
2019-04-08 09:32:05 +00:00
ModDisplay . Current . Value = mods ;
2018-04-13 09:19:50 +00:00
2020-07-22 03:41:06 +00:00
configVisibilityMode = config . GetBindable < HUDVisibilityMode > ( OsuSetting . HUDVisibilityMode ) ;
2019-12-12 06:05:29 +00:00
2020-07-22 03:41:06 +00:00
if ( configVisibilityMode . Value = = HUDVisibilityMode . Never & & ! hasShownNotificationOnce )
2019-12-12 06:05:29 +00:00
{
hasShownNotificationOnce = true ;
notificationOverlay ? . Post ( new SimpleNotification
{
2020-12-01 02:38:16 +00:00
Text = $"The score overlay is currently disabled. You can toggle this by pressing {config.LookupKeyBindings(GlobalAction.ToggleInGameInterface)}."
2019-12-12 06:05:29 +00:00
} ) ;
}
// start all elements hidden
hideTargets . ForEach ( d = > d . Hide ( ) ) ;
}
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." ) ;
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2020-10-15 07:56:05 +00:00
ShowHud . BindValueChanged ( visible = > hideTargets . ForEach ( d = > d . FadeTo ( visible . NewValue ? 1 : 0 , FADE_DURATION , FADE_EASING ) ) ) ;
2018-04-13 09:19:50 +00:00
2020-07-22 03:41:06 +00:00
IsBreakTime . BindValueChanged ( _ = > updateVisibility ( ) ) ;
configVisibilityMode . BindValueChanged ( _ = > updateVisibility ( ) , true ) ;
2018-04-13 09:19:50 +00:00
2019-05-10 06:39:25 +00:00
replayLoaded . BindValueChanged ( replayLoadedValueChanged , true ) ;
2018-04-13 09:19:50 +00:00
}
2020-10-14 09:51:53 +00:00
protected override void Update ( )
{
base . Update ( ) ;
2020-10-15 08:11:02 +00:00
2021-05-11 03:42:32 +00:00
Vector2 lowestScreenSpace = Vector2 . Zero ;
2021-05-11 08:48:08 +00:00
// LINQ cast can be removed when IDrawable interface includes Anchor / RelativeSizeAxes.
foreach ( var element in mainComponents . Components . Cast < Drawable > ( ) )
2021-05-11 03:42:32 +00:00
{
2021-05-11 08:48:08 +00:00
// for now align top-right components with the bottom-edge of the lowest top-anchored hud element.
if ( ! element . Anchor . HasFlagFast ( Anchor . TopRight ) & & ! element . RelativeSizeAxes . HasFlagFast ( Axes . X ) )
continue ;
2021-05-11 03:42:32 +00:00
2021-05-11 08:48:08 +00: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 )
continue ;
2020-10-15 09:30:44 +00:00
2021-05-11 08:48:08 +00:00
var bottomRight = element . ScreenSpaceDrawQuad . BottomRight ;
if ( bottomRight . Y > lowestScreenSpace . Y )
lowestScreenSpace = bottomRight ;
2021-05-11 03:42:32 +00:00
}
2021-05-11 08:48:08 +00:00
topRightElements . Y = TopScoringElementsHeight = ToLocalSpace ( lowestScreenSpace ) . Y ;
bottomRightElements . Y = - Progress . Height ;
2020-10-14 09:51:53 +00:00
}
2020-07-22 03:41:06 +00:00
private void updateVisibility ( )
{
if ( ShowHud . Disabled )
return ;
2020-10-30 05:19:40 +00:00
if ( holdingForHUD )
{
ShowHud . Value = true ;
return ;
}
2020-07-22 03:41:06 +00:00
switch ( configVisibilityMode . Value )
{
case HUDVisibilityMode . Never :
ShowHud . Value = false ;
break ;
2020-10-20 05:19:04 +00:00
case HUDVisibilityMode . HideDuringGameplay :
// always show during replay as we want the seek bar to be visible.
ShowHud . Value = replayLoaded . Value | | IsBreakTime . Value ;
break ;
2020-07-22 03:41:06 +00:00
case HUDVisibilityMode . Always :
ShowHud . Value = true ;
break ;
}
}
2019-02-21 09:56:34 +00:00
private void replayLoadedValueChanged ( ValueChangedEvent < bool > e )
2018-04-13 09:19:50 +00:00
{
2019-02-21 09:56:34 +00:00
PlayerSettingsOverlay . ReplayLoaded = e . NewValue ;
2018-04-13 09:19:50 +00:00
2019-02-21 09:56:34 +00:00
if ( e . NewValue )
2018-04-13 09:19:50 +00:00
{
PlayerSettingsOverlay . Show ( ) ;
ModDisplay . FadeIn ( 200 ) ;
2018-10-30 12:32:12 +00:00
KeyCounter . Margin = new MarginPadding ( 10 ) { Bottom = 30 } ;
2018-04-13 09:19:50 +00:00
}
else
{
PlayerSettingsOverlay . Hide ( ) ;
ModDisplay . Delay ( 2000 ) . FadeOut ( 200 ) ;
2018-10-30 12:32:12 +00:00
KeyCounter . Margin = new MarginPadding ( 10 ) ;
2018-04-13 09:19:50 +00:00
}
2020-07-22 03:41:06 +00:00
updateVisibility ( ) ;
2018-04-13 09:19:50 +00:00
}
2019-03-19 14:44:15 +00:00
protected virtual void BindDrawableRuleset ( DrawableRuleset drawableRuleset )
2018-04-13 09:19:50 +00:00
{
2019-03-19 14:44:15 +00:00
( drawableRuleset as ICanAttachKeyCounter ) ? . Attach ( KeyCounter ) ;
2018-04-13 09:19:50 +00:00
2019-03-19 14:44:15 +00:00
replayLoaded . BindTo ( drawableRuleset . HasReplayLoaded ) ;
2018-04-13 09:19:50 +00:00
2019-03-20 05:49:33 +00:00
Progress . BindDrawableRuleset ( drawableRuleset ) ;
2018-04-13 09:19:50 +00:00
}
2021-05-06 06:16:16 +00:00
protected FailingLayer CreateFailingLayer ( ) = > new FailingLayer
2020-06-26 17:22:30 +00:00
{
ShowHealth = { BindTarget = ShowHealthbar }
} ;
2020-04-09 05:31:25 +00:00
2021-05-06 06:16:16 +00:00
protected KeyCounterDisplay CreateKeyCounter ( ) = > new KeyCounterDisplay
2018-04-13 09:19:50 +00:00
{
Anchor = Anchor . BottomRight ,
Origin = Anchor . BottomRight ,
} ;
2021-05-06 06:16:16 +00:00
protected SongProgress CreateProgress ( ) = > new SongProgress
2018-04-13 09:19:50 +00:00
{
Anchor = Anchor . BottomLeft ,
Origin = Anchor . BottomLeft ,
RelativeSizeAxes = Axes . X ,
} ;
2021-05-06 06:16:16 +00:00
protected HoldForMenuButton CreateHoldForMenuButton ( ) = > new HoldForMenuButton
2018-04-21 15:24:31 +00:00
{
Anchor = Anchor . BottomRight ,
Origin = Anchor . BottomRight ,
} ;
2021-05-06 06:16:16 +00:00
protected ModDisplay CreateModsContainer ( ) = > new ModDisplay
2018-04-13 09:19:50 +00:00
{
Anchor = Anchor . TopRight ,
Origin = Anchor . TopRight ,
AutoSizeAxes = Axes . Both ,
} ;
2021-05-06 06:16:16 +00:00
protected PlayerSettingsOverlay CreatePlayerSettingsOverlay ( ) = > new PlayerSettingsOverlay ( ) ;
2018-04-13 09:19:50 +00:00
2020-10-30 05:19:40 +00:00
public bool OnPressed ( GlobalAction action )
{
switch ( action )
{
case GlobalAction . HoldForHUD :
holdingForHUD = true ;
updateVisibility ( ) ;
return true ;
2020-11-30 01:59:02 +00: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 05:00:54 +00:00
2020-11-30 01:59:02 +00:00
return true ;
2020-10-30 05:19:40 +00:00
}
return false ;
}
public void OnReleased ( GlobalAction action )
{
switch ( action )
{
case GlobalAction . HoldForHUD :
holdingForHUD = false ;
updateVisibility ( ) ;
break ;
}
}
2018-04-13 09:19:50 +00:00
}
}