2018-01-05 11:21:19 +00:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
2017-05-16 03:53:50 +00:00
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System ;
using System.Collections.Generic ;
using osu.Framework.Allocation ;
using osu.Framework.Configuration ;
2018-01-18 07:57:32 +00:00
using osu.Framework.Configuration.Tracking ;
2017-05-16 03:53:50 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2017-06-20 05:54:23 +00:00
using osu.Framework.Graphics.Shapes ;
2017-05-16 03:53:50 +00:00
using osu.Framework.Graphics.Sprites ;
using osu.Game.Graphics ;
using OpenTK ;
using OpenTK.Graphics ;
using osu.Framework.Extensions.Color4Extensions ;
2017-12-25 15:44:35 +00:00
using osu.Game.Graphics.Sprites ;
2017-05-16 03:53:50 +00:00
namespace osu.Game.Overlays
{
public class OnScreenDisplay : Container
{
private readonly Container box ;
2018-01-07 20:40:00 +00:00
public override bool HandleKeyboardInput = > false ;
public override bool HandleMouseInput = > false ;
2017-05-16 03:53:50 +00:00
private readonly SpriteText textLine1 ;
private readonly SpriteText textLine2 ;
private readonly SpriteText textLine3 ;
private const float height = 110 ;
private const float height_contracted = height * 0.9f ;
private readonly FillFlowContainer < OptionLight > optionLights ;
public OnScreenDisplay ( )
{
RelativeSizeAxes = Axes . Both ;
Children = new Drawable [ ]
{
box = new Container
{
Origin = Anchor . Centre ,
RelativePositionAxes = Axes . Both ,
Position = new Vector2 ( 0.5f , 0.75f ) ,
Masking = true ,
AutoSizeAxes = Axes . X ,
Height = height_contracted ,
Alpha = 0 ,
CornerRadius = 20 ,
Children = new Drawable [ ]
{
new Box
{
RelativeSizeAxes = Axes . Both ,
Colour = Color4 . Black ,
Alpha = 0.7f ,
} ,
new Container // purely to add a minimum width
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
Width = 240 ,
RelativeSizeAxes = Axes . Y ,
} ,
2017-12-25 15:44:35 +00:00
textLine1 = new OsuSpriteText
2017-05-16 03:53:50 +00:00
{
Padding = new MarginPadding ( 10 ) ,
Font = @"Exo2.0-Black" ,
Spacing = new Vector2 ( 1 , 0 ) ,
TextSize = 14 ,
Anchor = Anchor . TopCentre ,
Origin = Anchor . TopCentre ,
} ,
2017-12-25 15:44:35 +00:00
textLine2 = new OsuSpriteText
2017-05-16 03:53:50 +00:00
{
TextSize = 24 ,
Font = @"Exo2.0-Light" ,
2017-05-19 13:49:47 +00:00
Padding = new MarginPadding { Left = 10 , Right = 10 } ,
2017-05-16 03:53:50 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . BottomCentre ,
} ,
new FillFlowContainer
{
Anchor = Anchor . BottomCentre ,
Origin = Anchor . BottomCentre ,
AutoSizeAxes = Axes . Both ,
Direction = FillDirection . Vertical ,
Children = new Drawable [ ]
{
optionLights = new FillFlowContainer < OptionLight >
{
Padding = new MarginPadding { Top = 20 , Bottom = 5 } ,
Spacing = new Vector2 ( 5 , 0 ) ,
Direction = FillDirection . Horizontal ,
Anchor = Anchor . TopCentre ,
Origin = Anchor . TopCentre ,
AutoSizeAxes = Axes . Both
} ,
2017-12-25 15:44:35 +00:00
textLine3 = new OsuSpriteText
2017-05-16 03:53:50 +00:00
{
Padding = new MarginPadding { Bottom = 15 } ,
Font = @"Exo2.0-Bold" ,
TextSize = 12 ,
Alpha = 0.3f ,
Anchor = Anchor . TopCentre ,
Origin = Anchor . TopCentre ,
} ,
}
}
}
} ,
} ;
}
[BackgroundDependencyLoader]
private void load ( FrameworkConfigManager frameworkConfig )
{
2018-01-18 09:45:48 +00:00
BeginTracking ( this , frameworkConfig ) ;
2018-01-18 06:26:03 +00:00
}
2017-05-16 03:53:50 +00:00
2018-01-18 09:45:48 +00:00
private readonly Dictionary < ( object , IConfigManager ) , TrackedSettings > trackedConfigManagers = new Dictionary < ( object , IConfigManager ) , TrackedSettings > ( ) ;
2017-05-16 03:53:50 +00:00
2018-01-18 09:45:48 +00:00
/// <summary>
/// Registers a <see cref="ConfigManager{T}"/> to have its settings tracked by this <see cref="OnScreenDisplay"/>.
/// </summary>
/// <param name="source">The object that is registering the <see cref="ConfigManager{T}"/> to be tracked.</param>
/// <param name="configManager">The <see cref="ConfigManager{T}"/> to be tracked.</param>
/// <exception cref="ArgumentNullException">If <paramref name="configManager"/> is null.</exception>
/// <exception cref="InvalidOperationException">If <paramref name="configManager"/> is already being tracked from the same <paramref name="source"/>.</exception>
public void BeginTracking ( object source , ITrackableConfigManager configManager )
2018-01-18 06:26:03 +00:00
{
if ( configManager = = null ) throw new ArgumentNullException ( nameof ( configManager ) ) ;
2017-05-16 03:53:50 +00:00
2018-01-18 08:40:05 +00:00
if ( trackedConfigManagers . ContainsKey ( ( source , configManager ) ) )
2018-01-18 09:45:48 +00:00
throw new InvalidOperationException ( $"{nameof(configManager)} is already registered." ) ;
2017-06-08 08:58:28 +00:00
2018-01-18 06:26:03 +00:00
var trackedSettings = configManager . CreateTrackedSettings ( ) ;
if ( trackedSettings = = null )
return ;
2017-05-16 03:53:50 +00:00
2018-01-18 07:57:32 +00:00
configManager . LoadInto ( trackedSettings ) ;
2018-01-18 06:26:03 +00:00
trackedSettings . SettingChanged + = display ;
2017-05-16 03:53:50 +00:00
2018-01-18 08:40:05 +00:00
trackedConfigManagers . Add ( ( source , configManager ) , trackedSettings ) ;
2018-01-18 06:26:03 +00:00
}
2018-01-18 09:45:48 +00:00
/// <summary>
/// Unregisters a <see cref="ConfigManager{T}"/> from having its settings tracked by this <see cref="OnScreenDisplay"/>.
/// </summary>
/// <param name="source">The object that registered the <see cref="ConfigManager{T}"/> to be tracked.</param>
/// <param name="configManager">The <see cref="ConfigManager{T}"/> that is being tracked.</param>
/// <exception cref="ArgumentNullException">If <paramref name="configManager"/> is null.</exception>
/// <exception cref="InvalidOperationException">If <paramref name="configManager"/> is not being tracked from the same <see cref="source"/>.</exception>
public void StopTracking ( object source , ITrackableConfigManager configManager )
2017-05-16 03:53:50 +00:00
{
2018-01-18 06:26:03 +00:00
if ( configManager = = null ) throw new ArgumentNullException ( nameof ( configManager ) ) ;
2018-01-18 08:40:05 +00:00
if ( ! trackedConfigManagers . TryGetValue ( ( source , configManager ) , out var existing ) )
2018-01-18 09:45:48 +00:00
throw new InvalidOperationException ( $"{nameof(configManager)} is not registered." ) ;
2018-01-18 06:26:03 +00:00
existing . Unload ( ) ;
existing . SettingChanged - = display ;
2017-05-16 03:53:50 +00:00
2018-01-18 08:40:05 +00:00
trackedConfigManagers . Remove ( ( source , configManager ) ) ;
2017-05-16 03:53:50 +00:00
}
2018-01-18 06:26:03 +00:00
private void display ( SettingDescription description )
2017-05-16 03:53:50 +00:00
{
Schedule ( ( ) = >
{
2018-01-18 06:26:03 +00:00
textLine1 . Text = description . Name . ToUpper ( ) ;
textLine2 . Text = description . Value ;
textLine3 . Text = description . Shortcut . ToUpper ( ) ;
2017-05-16 03:53:50 +00:00
2017-07-16 14:37:59 +00:00
box . Animate (
2017-07-22 18:50:25 +00:00
b = > b . FadeIn ( 500 , Easing . OutQuint ) ,
b = > b . ResizeHeightTo ( height , 500 , Easing . OutQuint )
2017-07-16 14:37:59 +00:00
) . Then (
2017-07-22 18:50:25 +00:00
b = > b . FadeOutFromOne ( 1500 , Easing . InQuint ) ,
b = > b . ResizeHeightTo ( height_contracted , 1500 , Easing . InQuint )
2017-07-16 14:37:59 +00:00
) ;
2017-05-16 03:53:50 +00:00
int optionCount = 0 ;
int selectedOption = - 1 ;
2018-01-18 06:26:03 +00:00
if ( description . RawValue is bool )
2017-05-16 03:53:50 +00:00
{
optionCount = 1 ;
2018-01-18 06:26:03 +00:00
if ( ( bool ) description . RawValue ) selectedOption = 0 ;
2017-05-16 03:53:50 +00:00
}
2018-01-18 06:26:03 +00:00
else if ( description . RawValue is Enum )
2017-05-16 03:53:50 +00:00
{
2018-01-18 06:26:03 +00:00
var values = Enum . GetValues ( description . RawValue . GetType ( ) ) ;
2017-05-16 03:53:50 +00:00
optionCount = values . Length ;
2018-01-18 06:26:03 +00:00
selectedOption = Convert . ToInt32 ( description . RawValue ) ;
2017-05-16 03:53:50 +00:00
}
textLine2 . Origin = optionCount > 0 ? Anchor . BottomCentre : Anchor . Centre ;
textLine2 . Y = optionCount > 0 ? 0 : 5 ;
2017-06-25 05:46:59 +00:00
if ( optionLights . Children . Count ! = optionCount )
2017-05-16 03:53:50 +00:00
{
optionLights . Clear ( ) ;
for ( int i = 0 ; i < optionCount ; i + + )
optionLights . Add ( new OptionLight ( ) ) ;
}
for ( int i = 0 ; i < optionCount ; i + + )
2017-06-25 05:46:59 +00:00
optionLights . Children [ i ] . Glowing = i = = selectedOption ;
2017-05-16 03:53:50 +00:00
} ) ;
}
private class OptionLight : Container
{
private Color4 glowingColour , idleColour ;
private const float transition_speed = 300 ;
private const float glow_strength = 0.4f ;
private readonly Box fill ;
public OptionLight ( )
{
Children = new [ ]
{
fill = new Box
{
RelativeSizeAxes = Axes . Both ,
Alpha = 1 ,
} ,
} ;
}
private bool glowing ;
public bool Glowing
{
set
{
glowing = value ;
if ( ! IsLoaded ) return ;
updateGlow ( ) ;
}
}
private void updateGlow ( )
{
if ( glowing )
{
2017-07-22 18:50:25 +00:00
fill . FadeColour ( glowingColour , transition_speed , Easing . OutQuint ) ;
FadeEdgeEffectTo ( glow_strength , transition_speed , Easing . OutQuint ) ;
2017-05-16 03:53:50 +00:00
}
else
{
2017-07-22 18:50:25 +00:00
FadeEdgeEffectTo ( 0 , transition_speed , Easing . OutQuint ) ;
fill . FadeColour ( idleColour , transition_speed , Easing . OutQuint ) ;
2017-05-16 03:53:50 +00:00
}
}
[BackgroundDependencyLoader]
private void load ( OsuColour colours )
{
fill . Colour = idleColour = Color4 . White . Opacity ( 0.4f ) ;
glowingColour = Color4 . White ;
Size = new Vector2 ( 25 , 5 ) ;
Masking = true ;
CornerRadius = 3 ;
2017-06-12 03:48:47 +00:00
EdgeEffect = new EdgeEffectParameters
2017-05-16 03:53:50 +00:00
{
Colour = colours . BlueDark . Opacity ( glow_strength ) ,
Type = EdgeEffectType . Glow ,
Radius = 8 ,
} ;
2017-08-17 16:46:16 +00:00
}
2017-05-16 03:53:50 +00:00
2017-08-17 16:46:16 +00:00
protected override void LoadComplete ( )
{
2017-05-16 03:53:50 +00:00
updateGlow ( ) ;
2017-07-21 15:24:09 +00:00
FinishTransforms ( true ) ;
2017-05-16 03:53:50 +00:00
}
}
}
}