2020-03-17 15:59:34 +08: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.
using System ;
using osu.Framework ;
using osu.Framework.Allocation ;
using osu.Framework.Extensions.Color4Extensions ;
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Colour ;
using osu.Framework.Graphics.Containers ;
using osu.Framework.Graphics.Shapes ;
2020-05-20 22:46:47 +08:00
using osu.Framework.Input.Events ;
2020-03-17 15:59:34 +08:00
using osu.Game.Scoring ;
2020-05-16 17:17:32 +08:00
using osu.Game.Screens.Ranking.Contracted ;
2020-03-17 15:59:34 +08:00
using osu.Game.Screens.Ranking.Expanded ;
2020-07-22 10:29:23 +08:00
using osu.Game.Users ;
2020-03-17 15:59:34 +08:00
using osuTK ;
using osuTK.Graphics ;
namespace osu.Game.Screens.Ranking
{
public class ScorePanel : CompositeDrawable , IStateful < PanelState >
{
/// <summary>
/// Width of the panel when contracted.
/// </summary>
2020-05-21 16:47:14 +08:00
public const float CONTRACTED_WIDTH = 130 ;
2020-03-17 15:59:34 +08:00
/// <summary>
/// Height of the panel when contracted.
/// </summary>
2020-10-09 12:26:09 +08:00
private const float contracted_height = 385 ;
2020-03-17 15:59:34 +08:00
/// <summary>
/// Width of the panel when expanded.
/// </summary>
2020-05-01 13:13:38 +08:00
public const float EXPANDED_WIDTH = 360 ;
2020-03-17 15:59:34 +08:00
/// <summary>
/// Height of the panel when expanded.
/// </summary>
2020-10-07 15:30:26 +08:00
private const float expanded_height = 586 ;
2020-03-17 15:59:34 +08:00
/// <summary>
/// Height of the top layer when the panel is expanded.
/// </summary>
private const float expanded_top_layer_height = 53 ;
/// <summary>
/// Height of the top layer when the panel is contracted.
/// </summary>
2020-05-21 16:47:14 +08:00
private const float contracted_top_layer_height = 30 ;
2020-03-17 15:59:34 +08:00
/// <summary>
/// Duration for the panel to resize into its expanded/contracted size.
/// </summary>
private const double resize_duration = 200 ;
/// <summary>
/// Delay after <see cref="resize_duration"/> before the top layer is expanded.
/// </summary>
private const double top_layer_expand_delay = 100 ;
/// <summary>
/// Duration for the top layer expansion.
/// </summary>
private const double top_layer_expand_duration = 200 ;
/// <summary>
/// Duration for the panel contents to fade in.
/// </summary>
private const double content_fade_duration = 50 ;
private static readonly ColourInfo expanded_top_layer_colour = ColourInfo . GradientVertical ( Color4Extensions . FromHex ( "#444" ) , Color4Extensions . FromHex ( "#333" ) ) ;
private static readonly ColourInfo expanded_middle_layer_colour = ColourInfo . GradientVertical ( Color4Extensions . FromHex ( "#555" ) , Color4Extensions . FromHex ( "#333" ) ) ;
private static readonly Color4 contracted_top_layer_colour = Color4Extensions . FromHex ( "#353535" ) ;
2020-05-16 17:17:32 +08:00
private static readonly Color4 contracted_middle_layer_colour = Color4Extensions . FromHex ( "#353535" ) ;
2020-03-17 15:59:34 +08:00
public event Action < PanelState > StateChanged ;
2020-06-18 15:50:45 +08:00
/// <summary>
2020-06-19 20:41:48 +08:00
/// An action to be invoked if this <see cref="ScorePanel"/> is clicked while in an expanded state.
2020-06-18 15:50:45 +08:00
/// </summary>
2020-06-19 20:41:48 +08:00
public Action PostExpandAction ;
2020-06-18 15:50:45 +08:00
2020-05-20 22:46:47 +08:00
public readonly ScoreInfo Score ;
2020-03-17 15:59:34 +08:00
2020-05-21 21:07:06 +08:00
private Container content ;
2020-03-17 15:59:34 +08:00
private Container topLayerContainer ;
private Drawable topLayerBackground ;
private Container topLayerContentContainer ;
private Drawable topLayerContent ;
private Container middleLayerContainer ;
private Drawable middleLayerBackground ;
private Container middleLayerContentContainer ;
private Drawable middleLayerContent ;
public ScorePanel ( ScoreInfo score )
{
2020-05-20 22:46:47 +08:00
Score = score ;
2020-03-17 15:59:34 +08:00
}
[BackgroundDependencyLoader]
private void load ( )
{
2020-10-07 15:30:26 +08:00
// ScorePanel doesn't include the top extruding area in its own size.
// Adding a manual offset here allows the expanded version to take on an "acceptable" vertical centre when at 100% UI scale.
const float vertical_fudge = 20 ;
2020-05-21 21:07:06 +08:00
InternalChild = content = new Container
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2020-05-26 15:27:41 +08:00
Size = new Vector2 ( 40 ) ,
2020-10-07 15:30:26 +08:00
Y = vertical_fudge ,
2020-05-21 21:07:06 +08:00
Children = new Drawable [ ]
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
topLayerContainer = new Container
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
Name = "Top layer" ,
RelativeSizeAxes = Axes . X ,
2020-05-26 15:27:41 +08:00
Alpha = 0 ,
2020-05-21 21:07:06 +08:00
Height = 120 ,
Children = new Drawable [ ]
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
new Container
{
RelativeSizeAxes = Axes . Both ,
CornerRadius = 20 ,
CornerExponent = 2.5f ,
Masking = true ,
Child = topLayerBackground = new Box { RelativeSizeAxes = Axes . Both }
} ,
topLayerContentContainer = new Container { RelativeSizeAxes = Axes . Both }
}
} ,
middleLayerContainer = new Container
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
Name = "Middle layer" ,
RelativeSizeAxes = Axes . Both ,
Children = new Drawable [ ]
2020-03-17 15:59:34 +08:00
{
2020-05-21 21:07:06 +08:00
new Container
{
RelativeSizeAxes = Axes . Both ,
CornerRadius = 20 ,
CornerExponent = 2.5f ,
Masking = true ,
2020-07-22 10:29:23 +08:00
Children = new [ ]
{
middleLayerBackground = new Box { RelativeSizeAxes = Axes . Both } ,
new UserCoverBackground
{
RelativeSizeAxes = Axes . Both ,
User = Score . User ,
Colour = ColourInfo . GradientVertical ( Color4 . White . Opacity ( 0.5f ) , Color4Extensions . FromHex ( "#444" ) . Opacity ( 0 ) )
2020-07-28 19:50:55 +08:00
}
2020-07-22 10:29:23 +08:00
}
2020-05-21 21:07:06 +08:00
} ,
middleLayerContentContainer = new Container { RelativeSizeAxes = Axes . Both }
}
2020-03-17 15:59:34 +08:00
}
}
} ;
}
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
updateState ( ) ;
2020-07-22 10:29:23 +08:00
topLayerBackground . FinishTransforms ( false , nameof ( Colour ) ) ;
middleLayerBackground . FinishTransforms ( false , nameof ( Colour ) ) ;
2020-03-17 15:59:34 +08:00
}
private PanelState state = PanelState . Contracted ;
public PanelState State
{
get = > state ;
set
{
if ( state = = value )
return ;
state = value ;
if ( LoadState > = LoadState . Ready )
updateState ( ) ;
StateChanged ? . Invoke ( value ) ;
}
}
private void updateState ( )
{
topLayerContent ? . FadeOut ( content_fade_duration ) . Expire ( ) ;
middleLayerContent ? . FadeOut ( content_fade_duration ) . Expire ( ) ;
switch ( state )
{
case PanelState . Expanded :
2020-05-21 21:07:06 +08:00
Size = new Vector2 ( EXPANDED_WIDTH , expanded_height ) ;
2020-03-17 15:59:34 +08:00
topLayerBackground . FadeColour ( expanded_top_layer_colour , resize_duration , Easing . OutQuint ) ;
middleLayerBackground . FadeColour ( expanded_middle_layer_colour , resize_duration , Easing . OutQuint ) ;
2020-07-31 22:11:42 +08:00
topLayerContentContainer . Add ( topLayerContent = new ExpandedPanelTopContent ( Score . User ) . With ( d = > d . Alpha = 0 ) ) ;
middleLayerContentContainer . Add ( middleLayerContent = new ExpandedPanelMiddleContent ( Score ) . With ( d = > d . Alpha = 0 ) ) ;
2020-03-17 15:59:34 +08:00
break ;
case PanelState . Contracted :
2020-05-21 21:07:06 +08:00
Size = new Vector2 ( CONTRACTED_WIDTH , contracted_height ) ;
2020-03-17 15:59:34 +08:00
topLayerBackground . FadeColour ( contracted_top_layer_colour , resize_duration , Easing . OutQuint ) ;
middleLayerBackground . FadeColour ( contracted_middle_layer_colour , resize_duration , Easing . OutQuint ) ;
2020-05-16 17:17:32 +08:00
2020-07-31 22:11:42 +08:00
topLayerContentContainer . Add ( topLayerContent = new ContractedPanelTopContent ( Score ) . With ( d = > d . Alpha = 0 ) ) ;
middleLayerContentContainer . Add ( middleLayerContent = new ContractedPanelMiddleContent ( Score ) . With ( d = > d . Alpha = 0 ) ) ;
2020-03-17 15:59:34 +08:00
break ;
}
2020-05-21 21:07:06 +08:00
content . ResizeTo ( Size , resize_duration , Easing . OutQuint ) ;
2020-05-21 17:07:31 +08:00
bool topLayerExpanded = topLayerContainer . Y < 0 ;
// If the top layer was already expanded, then we don't need to wait for the resize and can instead transform immediately. This looks better when changing the panel state.
using ( BeginDelayedSequence ( topLayerExpanded ? 0 : resize_duration + top_layer_expand_delay , true ) )
2020-03-17 15:59:34 +08:00
{
2020-05-26 15:27:41 +08:00
topLayerContainer . FadeIn ( ) ;
2020-03-17 15:59:34 +08:00
switch ( state )
{
case PanelState . Expanded :
topLayerContainer . MoveToY ( - expanded_top_layer_height / 2 , top_layer_expand_duration , Easing . OutQuint ) ;
middleLayerContainer . MoveToY ( expanded_top_layer_height / 2 , top_layer_expand_duration , Easing . OutQuint ) ;
break ;
case PanelState . Contracted :
topLayerContainer . MoveToY ( - contracted_top_layer_height / 2 , top_layer_expand_duration , Easing . OutQuint ) ;
middleLayerContainer . MoveToY ( contracted_top_layer_height / 2 , top_layer_expand_duration , Easing . OutQuint ) ;
break ;
}
topLayerContent ? . FadeIn ( content_fade_duration ) ;
middleLayerContent ? . FadeIn ( content_fade_duration ) ;
}
}
2020-05-20 22:46:47 +08:00
2020-06-18 15:50:45 +08:00
public override Vector2 Size
{
get = > base . Size ;
set
{
base . Size = value ;
2020-06-19 20:41:48 +08:00
// Auto-size isn't used to avoid 1-frame issues and because the score panel is removed/re-added to the container.
2020-06-19 16:28:35 +08:00
if ( trackingContainer ! = null )
trackingContainer . Size = value ;
2020-06-18 15:50:45 +08:00
}
}
2020-05-20 22:46:47 +08:00
protected override bool OnClick ( ClickEvent e )
{
if ( State = = PanelState . Contracted )
2020-06-18 15:50:45 +08:00
{
2020-06-19 20:41:48 +08:00
State = PanelState . Expanded ;
2020-06-18 15:50:45 +08:00
return true ;
}
PostExpandAction ? . Invoke ( ) ;
2020-05-20 22:46:47 +08:00
return true ;
}
2020-06-04 20:48:55 +08:00
public override bool ReceivePositionalInputAt ( Vector2 screenSpacePos )
= > base . ReceivePositionalInputAt ( screenSpacePos )
| | topLayerContainer . ReceivePositionalInputAt ( screenSpacePos )
| | middleLayerContainer . ReceivePositionalInputAt ( screenSpacePos ) ;
2020-06-17 21:29:00 +08:00
2020-06-19 16:28:35 +08:00
private ScorePanelTrackingContainer trackingContainer ;
2020-06-17 21:29:00 +08:00
2020-06-19 20:41:48 +08:00
/// <summary>
/// Creates a <see cref="ScorePanelTrackingContainer"/> which this <see cref="ScorePanel"/> can reside inside.
/// The <see cref="ScorePanelTrackingContainer"/> will track the size of this <see cref="ScorePanel"/>.
/// </summary>
/// <remarks>
/// This <see cref="ScorePanel"/> is immediately added as a child of the <see cref="ScorePanelTrackingContainer"/>.
/// </remarks>
/// <returns>The <see cref="ScorePanelTrackingContainer"/>.</returns>
/// <exception cref="InvalidOperationException">If a <see cref="ScorePanelTrackingContainer"/> already exists.</exception>
2020-06-19 16:28:35 +08:00
public ScorePanelTrackingContainer CreateTrackingContainer ( )
2020-06-17 21:29:00 +08:00
{
2020-06-19 16:28:35 +08:00
if ( trackingContainer ! = null )
throw new InvalidOperationException ( "A score panel container has already been created." ) ;
2020-06-18 15:50:45 +08:00
2020-06-19 16:28:35 +08:00
return trackingContainer = new ScorePanelTrackingContainer ( this ) ;
2020-06-17 21:29:00 +08:00
}
2020-03-17 15:59:34 +08:00
}
}