2020-03-17 08:43:16 +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.
2020-03-18 09:28:42 +00:00
using System ;
2020-05-26 08:00:41 +00:00
using System.Collections.Generic ;
2020-06-19 08:28:35 +00:00
using System.Linq ;
2020-03-17 08:43:16 +00:00
using osu.Framework.Allocation ;
2020-05-28 12:40:01 +00:00
using osu.Framework.Bindables ;
2020-03-17 08:43:16 +00:00
using osu.Framework.Extensions.Color4Extensions ;
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
using osu.Framework.Graphics.Shapes ;
2020-10-11 01:30:13 +00:00
using osu.Framework.Input.Bindings ;
2021-09-16 09:26:12 +00:00
using osu.Framework.Input.Events ;
2020-03-17 08:43:16 +00:00
using osu.Framework.Screens ;
2021-06-09 08:18:52 +00:00
using osu.Game.Graphics ;
2020-03-17 13:21:16 +00:00
using osu.Game.Graphics.Containers ;
2020-03-17 08:43:16 +00:00
using osu.Game.Graphics.UserInterface ;
2020-10-11 01:30:13 +00:00
using osu.Game.Input.Bindings ;
2020-05-16 10:00:20 +00:00
using osu.Game.Online.API ;
2020-03-17 08:43:16 +00:00
using osu.Game.Scoring ;
2020-03-17 08:45:25 +00:00
using osu.Game.Screens.Play ;
2020-06-16 08:49:43 +00:00
using osu.Game.Screens.Ranking.Statistics ;
2020-03-17 08:43:16 +00:00
using osuTK ;
namespace osu.Game.Screens.Ranking
{
2021-01-04 09:24:21 +00:00
public abstract class ResultsScreen : ScreenWithBeatmapBackground , IKeyBindingHandler < GlobalAction >
2020-03-17 08:43:16 +00:00
{
protected const float BACKGROUND_BLUR = 20 ;
2020-06-17 10:28:40 +00:00
private static readonly float screen_height = 768 - TwoLayerButton . SIZE_EXTENDED . Y ;
2020-03-17 08:43:16 +00:00
public override bool DisallowExternalBeatmapRulesetChanges = > true ;
// Temporary for now to stop dual transitions. Should respect the current toolbar mode, but there's no way to do so currently.
public override bool HideOverlaysOnEnter = > true ;
2020-05-28 12:40:01 +00:00
public readonly Bindable < ScoreInfo > SelectedScore = new Bindable < ScoreInfo > ( ) ;
public readonly ScoreInfo Score ;
2020-07-31 10:57:05 +00:00
protected ScorePanelList ScorePanelList { get ; private set ; }
2020-05-28 12:40:01 +00:00
2021-08-13 07:14:23 +00:00
protected VerticalScrollContainer VerticalScrollContent { get ; private set ; }
2020-03-17 08:45:25 +00:00
[Resolved(CanBeNull = true)]
private Player player { get ; set ; }
2020-05-16 10:00:20 +00:00
[Resolved]
private IAPIProvider api { get ; set ; }
2020-06-18 07:50:45 +00:00
private StatisticsPanel statisticsPanel ;
2020-03-17 08:43:16 +00:00
private Drawable bottomPanel ;
2020-06-19 08:28:35 +00:00
private Container < ScorePanel > detachedPanelContainer ;
2020-03-17 08:43:16 +00:00
2021-09-07 06:18:59 +00:00
private bool lastFetchCompleted ;
2020-07-28 11:58:13 +00:00
2020-07-31 10:57:05 +00:00
private readonly bool allowRetry ;
2020-12-20 15:14:54 +00:00
private readonly bool allowWatchingReplay ;
2020-07-31 10:57:05 +00:00
2020-12-20 15:14:54 +00:00
protected ResultsScreen ( ScoreInfo score , bool allowRetry , bool allowWatchingReplay = true )
2020-03-17 08:43:16 +00:00
{
2020-03-29 14:50:16 +00:00
Score = score ;
2020-03-29 14:52:50 +00:00
this . allowRetry = allowRetry ;
2020-12-20 15:14:54 +00:00
this . allowWatchingReplay = allowWatchingReplay ;
2020-05-28 12:40:01 +00:00
SelectedScore . Value = score ;
2020-03-17 08:43:16 +00:00
}
[BackgroundDependencyLoader]
private void load ( )
{
2020-03-29 14:52:50 +00:00
FillFlowContainer buttons ;
2020-05-22 11:39:02 +00:00
InternalChild = new GridContainer
2020-03-17 08:43:16 +00:00
{
2020-05-22 11:39:02 +00:00
RelativeSizeAxes = Axes . Both ,
Content = new [ ]
2020-03-17 08:43:16 +00:00
{
2020-05-22 11:39:02 +00:00
new Drawable [ ]
2020-03-17 13:21:16 +00:00
{
2021-08-13 07:14:23 +00:00
VerticalScrollContent = new VerticalScrollContainer
2020-03-17 08:43:16 +00:00
{
2020-06-17 13:29:00 +00:00
RelativeSizeAxes = Axes . Both ,
2020-06-30 07:36:53 +00:00
ScrollbarVisible = false ,
Child = new Container
2020-05-22 11:39:02 +00:00
{
2020-06-30 07:36:53 +00:00
RelativeSizeAxes = Axes . Both ,
Children = new Drawable [ ]
2020-06-17 10:28:40 +00:00
{
2020-07-01 21:31:06 +00:00
statisticsPanel = new StatisticsPanel
{
RelativeSizeAxes = Axes . Both ,
Score = { BindTarget = SelectedScore }
} ,
2020-07-31 10:57:05 +00:00
ScorePanelList = new ScorePanelList
2020-06-17 10:28:40 +00:00
{
2020-06-30 07:36:53 +00:00
RelativeSizeAxes = Axes . Both ,
SelectedScore = { BindTarget = SelectedScore } ,
PostExpandAction = ( ) = > statisticsPanel . ToggleVisibility ( )
} ,
detachedPanelContainer = new Container < ScorePanel >
{
RelativeSizeAxes = Axes . Both
} ,
}
2020-05-22 11:39:02 +00:00
}
2020-06-30 07:36:53 +00:00
} ,
2020-05-22 11:39:02 +00:00
} ,
new [ ]
{
bottomPanel = new Container
2020-03-17 08:43:16 +00:00
{
2020-05-22 11:39:02 +00:00
Anchor = Anchor . BottomLeft ,
Origin = Anchor . BottomLeft ,
RelativeSizeAxes = Axes . X ,
Height = TwoLayerButton . SIZE_EXTENDED . Y ,
Alpha = 0 ,
2020-03-17 08:43:16 +00:00
Children = new Drawable [ ]
{
2020-05-22 11:39:02 +00:00
new Box
{
RelativeSizeAxes = Axes . Both ,
Colour = Color4Extensions . FromHex ( "#333" )
} ,
buttons = new FillFlowContainer
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
AutoSizeAxes = Axes . Both ,
Spacing = new Vector2 ( 5 ) ,
2020-12-20 15:14:54 +00:00
Direction = FillDirection . Horizontal
2020-05-22 11:39:02 +00:00
}
2020-03-17 08:43:16 +00:00
}
}
}
2020-05-22 11:39:02 +00:00
} ,
RowDimensions = new [ ]
{
new Dimension ( ) ,
new Dimension ( GridSizeMode . AutoSize )
2020-03-17 08:43:16 +00:00
}
} ;
2020-03-17 08:45:25 +00:00
2020-05-28 12:40:01 +00:00
if ( Score ! = null )
2020-11-20 05:32:23 +00:00
{
// only show flair / animation when arriving after watching a play that isn't autoplay.
2021-06-09 05:32:48 +00:00
bool shouldFlair = player ! = null & & Score . Mods . All ( m = > m . UserPlayable ) ;
2020-11-20 05:32:23 +00:00
ScorePanelList . AddScore ( Score , shouldFlair ) ;
}
2020-05-28 12:40:01 +00:00
2020-12-20 15:14:54 +00:00
if ( allowWatchingReplay )
{
buttons . Add ( new ReplayDownloadButton ( null )
{
Score = { BindTarget = SelectedScore } ,
Width = 300
} ) ;
}
2020-03-30 09:56:35 +00:00
if ( player ! = null & & allowRetry )
2020-03-17 08:45:25 +00:00
{
2020-03-30 09:56:35 +00:00
buttons . Add ( new RetryButton { Width = 300 } ) ;
2020-03-29 14:52:50 +00:00
2020-03-30 09:56:35 +00:00
AddInternal ( new HotkeyRetryOverlay
{
Action = ( ) = >
2020-03-17 08:45:25 +00:00
{
2020-03-30 09:56:35 +00:00
if ( ! this . IsCurrentScreen ( ) ) return ;
2020-03-17 08:45:25 +00:00
2020-03-30 09:56:35 +00:00
player ? . Restart ( ) ;
} ,
} ) ;
2020-03-17 08:45:25 +00:00
}
2020-03-17 08:43:16 +00:00
}
2020-05-16 10:00:20 +00:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2020-07-22 11:24:55 +00:00
var req = FetchScores ( fetchScoresCallback ) ;
2020-05-16 10:00:20 +00:00
2020-05-26 08:00:41 +00:00
if ( req ! = null )
api . Queue ( req ) ;
2020-06-18 13:27:27 +00:00
statisticsPanel . State . BindValueChanged ( onStatisticsStateChanged , true ) ;
2020-05-16 10:00:20 +00:00
}
2020-07-22 11:24:55 +00:00
protected override void Update ( )
{
base . Update ( ) ;
2021-09-07 06:18:59 +00:00
if ( lastFetchCompleted )
2020-07-22 11:24:55 +00:00
{
2021-09-07 06:18:59 +00:00
APIRequest nextPageRequest = null ;
2020-07-31 10:57:05 +00:00
if ( ScorePanelList . IsScrolledToStart )
2020-07-22 11:24:55 +00:00
nextPageRequest = FetchNextPage ( - 1 , fetchScoresCallback ) ;
2020-07-31 10:57:05 +00:00
else if ( ScorePanelList . IsScrolledToEnd )
2020-07-22 11:24:55 +00:00
nextPageRequest = FetchNextPage ( 1 , fetchScoresCallback ) ;
if ( nextPageRequest ! = null )
{
2021-09-07 06:18:59 +00:00
lastFetchCompleted = false ;
2020-07-22 11:24:55 +00:00
api . Queue ( nextPageRequest ) ;
}
}
}
2020-05-26 08:00:41 +00:00
/// <summary>
/// Performs a fetch/refresh of scores to be displayed.
/// </summary>
/// <param name="scoresCallback">A callback which should be called when fetching is completed. Scheduling is not required.</param>
/// <returns>An <see cref="APIRequest"/> responsible for the fetch operation. This will be queued and performed automatically.</returns>
protected virtual APIRequest FetchScores ( Action < IEnumerable < ScoreInfo > > scoresCallback ) = > null ;
2020-07-28 11:58:13 +00:00
/// <summary>
/// Performs a fetch of the next page of scores. This is invoked every frame until a non-null <see cref="APIRequest"/> is returned.
/// </summary>
/// <param name="direction">The fetch direction. -1 to fetch scores greater than the current start of the list, and 1 to fetch scores lower than the current end of the list.</param>
/// <param name="scoresCallback">A callback which should be called when fetching is completed. Scheduling is not required.</param>
/// <returns>An <see cref="APIRequest"/> responsible for the fetch operation. This will be queued and performed automatically.</returns>
2020-07-22 11:24:55 +00:00
protected virtual APIRequest FetchNextPage ( int direction , Action < IEnumerable < ScoreInfo > > scoresCallback ) = > null ;
private void fetchScoresCallback ( IEnumerable < ScoreInfo > scores ) = > Schedule ( ( ) = >
{
foreach ( var s in scores )
addScore ( s ) ;
2021-09-07 06:18:59 +00:00
lastFetchCompleted = true ;
2020-07-22 11:24:55 +00:00
} ) ;
2022-04-21 15:52:44 +00:00
public override void OnEntering ( ScreenTransitionEvent e )
2020-03-17 08:43:16 +00:00
{
2022-04-21 15:52:44 +00:00
base . OnEntering ( e ) ;
2020-03-17 08:43:16 +00:00
2021-01-04 09:32:23 +00:00
ApplyToBackground ( b = >
{
b . BlurAmount . Value = BACKGROUND_BLUR ;
2021-06-09 08:18:52 +00:00
b . FadeColour ( OsuColour . Gray ( 0.5f ) , 250 ) ;
2021-01-04 09:32:23 +00:00
} ) ;
2020-03-17 08:43:16 +00:00
bottomPanel . FadeTo ( 1 , 250 ) ;
}
2022-04-21 15:52:44 +00:00
public override bool OnExiting ( ScreenExitEvent e )
2020-07-14 07:00:43 +00:00
{
2022-04-21 15:52:44 +00:00
if ( base . OnExiting ( e ) )
2021-06-09 08:19:36 +00:00
return true ;
2020-07-14 07:00:43 +00:00
2021-06-09 08:19:36 +00:00
this . FadeOut ( 100 ) ;
return false ;
2020-07-14 07:00:43 +00:00
}
public override bool OnBackButton ( )
2020-03-17 08:43:16 +00:00
{
2020-06-18 13:27:27 +00:00
if ( statisticsPanel . State . Value = = Visibility . Visible )
{
statisticsPanel . Hide ( ) ;
return true ;
}
2020-07-14 07:00:43 +00:00
return false ;
2020-03-17 08:43:16 +00:00
}
2020-03-17 13:21:16 +00:00
2020-06-19 08:28:35 +00:00
private void addScore ( ScoreInfo score )
{
2020-07-31 10:57:05 +00:00
var panel = ScorePanelList . AddScore ( score ) ;
2020-06-19 08:28:35 +00:00
if ( detachedPanel ! = null )
panel . Alpha = 0 ;
}
private ScorePanel detachedPanel ;
2020-06-18 13:27:27 +00:00
private void onStatisticsStateChanged ( ValueChangedEvent < Visibility > state )
2020-03-17 13:21:16 +00:00
{
2020-06-19 08:28:35 +00:00
if ( state . NewValue = = Visibility . Visible )
2020-03-17 13:21:16 +00:00
{
2020-06-19 08:28:35 +00:00
// Detach the panel in its original location, and move into the desired location in the local container.
2020-07-31 10:57:05 +00:00
var expandedPanel = ScorePanelList . GetPanelForScore ( SelectedScore . Value ) ;
2020-06-19 08:28:35 +00:00
var screenSpacePos = expandedPanel . ScreenSpaceDrawQuad . TopLeft ;
2020-06-18 13:27:27 +00:00
2020-06-19 08:28:35 +00:00
// Detach and move into the local container.
2020-07-31 10:57:05 +00:00
ScorePanelList . Detach ( expandedPanel ) ;
2020-06-19 08:28:35 +00:00
detachedPanelContainer . Add ( expandedPanel ) ;
2020-06-22 06:42:55 +00:00
// Move into its original location in the local container first, then to the final location.
2021-10-27 04:04:41 +00:00
float origLocation = detachedPanelContainer . ToLocalSpace ( screenSpacePos ) . X ;
2020-09-24 03:49:32 +00:00
expandedPanel . MoveToX ( origLocation )
2020-06-22 06:42:55 +00:00
. Then ( )
2020-09-24 03:49:32 +00:00
. MoveToX ( StatisticsPanel . SIDE_PADDING , 150 , Easing . OutQuint ) ;
2020-06-19 08:28:35 +00:00
// Hide contracted panels.
2020-07-31 10:57:05 +00:00
foreach ( var contracted in ScorePanelList . GetScorePanels ( ) . Where ( p = > p . State = = PanelState . Contracted ) )
2020-06-19 08:28:35 +00:00
contracted . FadeOut ( 150 , Easing . OutQuint ) ;
2020-07-31 10:57:05 +00:00
ScorePanelList . HandleInput = false ;
2020-06-19 08:28:35 +00:00
// Dim background.
2021-06-10 06:26:57 +00:00
ApplyToBackground ( b = > b . FadeColour ( OsuColour . Gray ( 0.1f ) , 150 ) ) ;
2020-06-19 08:28:35 +00:00
detachedPanel = expandedPanel ;
2020-06-18 07:50:45 +00:00
}
2020-06-19 08:28:35 +00:00
else if ( detachedPanel ! = null )
2020-06-18 07:50:45 +00:00
{
2020-06-19 08:28:35 +00:00
var screenSpacePos = detachedPanel . ScreenSpaceDrawQuad . TopLeft ;
2020-06-18 13:27:27 +00:00
2020-06-19 08:28:35 +00:00
// Remove from the local container and re-attach.
detachedPanelContainer . Remove ( detachedPanel ) ;
2020-07-31 10:57:05 +00:00
ScorePanelList . Attach ( detachedPanel ) ;
2020-06-18 07:50:45 +00:00
2020-06-22 06:42:55 +00:00
// Move into its original location in the attached container first, then to the final location.
2021-11-18 07:15:51 +00:00
float origLocation = detachedPanel . Parent . ToLocalSpace ( screenSpacePos ) . X ;
detachedPanel . MoveToX ( origLocation )
2020-06-22 06:42:55 +00:00
. Then ( )
2021-11-18 07:15:51 +00:00
. MoveToX ( 0 , 150 , Easing . OutQuint ) ;
2020-06-19 08:28:35 +00:00
// Show contracted panels.
2020-07-31 10:57:05 +00:00
foreach ( var contracted in ScorePanelList . GetScorePanels ( ) . Where ( p = > p . State = = PanelState . Contracted ) )
2020-06-19 08:28:35 +00:00
contracted . FadeIn ( 150 , Easing . OutQuint ) ;
2020-07-31 10:57:05 +00:00
ScorePanelList . HandleInput = true ;
2020-06-19 08:28:35 +00:00
// Un-dim background.
2021-06-10 06:26:57 +00:00
ApplyToBackground ( b = > b . FadeColour ( OsuColour . Gray ( 0.5f ) , 150 ) ) ;
2020-06-19 08:28:35 +00:00
detachedPanel = null ;
2020-03-17 13:21:16 +00:00
}
}
2020-06-30 07:36:53 +00:00
2021-09-16 09:26:12 +00:00
public bool OnPressed ( KeyBindingPressEvent < GlobalAction > e )
2020-10-11 01:30:13 +00:00
{
2021-11-18 03:35:47 +00:00
if ( e . Repeat )
return false ;
2021-09-16 09:26:12 +00:00
switch ( e . Action )
2020-10-11 01:30:13 +00:00
{
case GlobalAction . Select :
statisticsPanel . ToggleVisibility ( ) ;
return true ;
}
return false ;
}
2021-09-16 09:26:12 +00:00
public void OnReleased ( KeyBindingReleaseEvent < GlobalAction > e )
2020-10-11 01:30:13 +00:00
{
}
2021-08-13 07:14:23 +00:00
protected class VerticalScrollContainer : OsuScrollContainer
2020-06-30 07:36:53 +00:00
{
protected override Container < Drawable > Content = > content ;
private readonly Container content ;
public VerticalScrollContainer ( )
{
2021-08-13 07:14:23 +00:00
Masking = false ;
2020-06-30 07:36:53 +00:00
base . Content . Add ( content = new Container { RelativeSizeAxes = Axes . X } ) ;
}
protected override void Update ( )
{
base . Update ( ) ;
content . Height = Math . Max ( screen_height , DrawHeight ) ;
}
}
2020-03-17 08:43:16 +00:00
}
}