2019-06-04 15:13:16 +08:00
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
2019-01-24 16:43:03 +08:00
// See the LICENCE file in the repository root for full licence text.
2018-04-13 17:19:50 +08:00
using System ;
2019-04-09 12:33:16 +08:00
using System.Collections.Generic ;
2020-03-24 13:13:46 +08:00
using System.IO ;
2018-04-13 17:19:50 +08:00
using System.Linq ;
2020-12-18 15:51:59 +08:00
using System.Threading.Tasks ;
2020-12-17 15:17:13 +08:00
using JetBrains.Annotations ;
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation ;
using osu.Framework.Audio ;
using osu.Framework.Audio.Sample ;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables ;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events ;
2018-04-13 17:19:50 +08:00
using osu.Framework.Logging ;
using osu.Framework.Screens ;
using osu.Framework.Threading ;
using osu.Game.Beatmaps ;
using osu.Game.Configuration ;
2019-01-04 12:29:37 +08:00
using osu.Game.Graphics.Containers ;
2020-03-24 13:13:46 +08:00
using osu.Game.IO.Archives ;
2018-04-13 17:19:50 +08:00
using osu.Game.Online.API ;
2018-06-06 14:10:09 +08:00
using osu.Game.Overlays ;
2020-12-18 17:31:49 +08:00
using osu.Game.Replays ;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets ;
using osu.Game.Rulesets.Mods ;
2020-12-18 17:31:49 +08:00
using osu.Game.Rulesets.Replays ;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Scoring ;
using osu.Game.Rulesets.UI ;
2018-11-28 15:12:57 +08:00
using osu.Game.Scoring ;
2020-03-24 13:13:46 +08:00
using osu.Game.Scoring.Legacy ;
2018-04-13 17:19:50 +08:00
using osu.Game.Screens.Ranking ;
using osu.Game.Skinning ;
2019-04-13 04:54:35 +08:00
using osu.Game.Users ;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Play
{
2019-11-01 14:32:06 +08:00
[Cached]
2020-10-14 18:39:48 +08:00
[Cached(typeof(ISamplePlaybackDisabler))]
public class Player : ScreenWithBeatmapBackground , ISamplePlaybackDisabler
2018-04-13 17:19:50 +08:00
{
2020-04-20 11:42:33 +08:00
/// <summary>
/// The delay upon completion of the beatmap before displaying the results screen.
/// </summary>
public const double RESULTS_DISPLAY_DELAY = 1000.0 ;
2019-06-25 15:55:49 +08:00
public override bool AllowBackButton = > false ; // handled by HoldForMenuButton
2018-04-13 17:19:50 +08:00
2019-06-12 15:33:15 +08:00
protected override UserActivity InitialActivity = > new UserActivity . SoloGame ( Beatmap . Value . BeatmapInfo , Ruleset . Value ) ;
2018-04-13 17:19:50 +08:00
2019-01-23 19:52:00 +08:00
public override float BackgroundParallaxAmount = > 0.1f ;
2018-04-13 17:19:50 +08:00
2019-01-28 14:41:54 +08:00
public override bool HideOverlaysOnEnter = > true ;
2018-04-13 17:19:50 +08:00
2020-08-31 17:16:13 +08:00
protected override OverlayActivation InitialOverlayActivationMode = > OverlayActivation . UserTriggered ;
2020-08-09 03:21:30 +08:00
2020-09-01 15:55:10 +08:00
// We are managing our own adjustments (see OnEntering/OnExiting).
public override bool AllowRateAdjustments = > false ;
2018-06-06 14:10:09 +08:00
2020-10-14 18:39:48 +08:00
private readonly Bindable < bool > samplePlaybackDisabled = new Bindable < bool > ( ) ;
2018-06-06 14:10:09 +08:00
2019-05-10 14:39:25 +08:00
/// <summary>
/// Whether gameplay should pause when the game window focus is lost.
/// </summary>
protected virtual bool PauseOnFocusLost = > true ;
2018-06-06 14:10:09 +08:00
2018-04-13 17:19:50 +08:00
public Action RestartRequested ;
public bool HasFailed { get ; private set ; }
private Bindable < bool > mouseWheelDisabled ;
2019-02-25 12:27:44 +08:00
private readonly Bindable < bool > storyboardReplacesBackground = new Bindable < bool > ( ) ;
2018-04-13 17:19:50 +08:00
2020-10-07 15:22:39 +08:00
protected readonly Bindable < bool > LocalUserPlaying = new Bindable < bool > ( ) ;
2020-10-06 20:09:35 +08:00
2018-04-13 17:19:50 +08:00
public int RestartCount ;
2018-11-29 13:56:29 +08:00
[Resolved]
private ScoreManager scoreManager { get ; set ; }
2018-04-13 17:19:50 +08:00
2019-08-26 11:21:49 +08:00
private RulesetInfo rulesetInfo ;
2018-04-13 17:19:50 +08:00
2019-08-26 11:21:49 +08:00
private Ruleset ruleset ;
2018-04-13 17:19:50 +08:00
2020-02-14 21:14:00 +08:00
[Resolved]
private IAPIProvider api { get ; set ; }
2018-04-13 17:19:50 +08:00
2020-09-01 17:07:19 +08:00
[Resolved]
private MusicController musicController { get ; set ; }
2020-08-03 03:34:35 +08:00
2018-04-13 17:19:50 +08:00
private SampleChannel sampleRestart ;
2019-12-12 14:14:59 +08:00
public BreakOverlay BreakOverlay ;
2018-04-13 17:19:50 +08:00
2020-10-11 06:15:20 +08:00
/// <summary>
/// Whether the gameplay is currently in a break.
/// </summary>
2020-10-11 20:46:55 +08:00
public readonly IBindable < bool > IsBreakTime = new BindableBool ( ) ;
2020-10-10 21:07:17 +08:00
2020-03-26 14:28:56 +08:00
private BreakTracker breakTracker ;
2020-05-13 03:12:48 +08:00
private SkipOverlay skipOverlay ;
2019-02-28 19:01:15 +08:00
protected ScoreProcessor ScoreProcessor { get ; private set ; }
2019-12-19 19:03:14 +08:00
protected HealthProcessor HealthProcessor { get ; private set ; }
2019-03-19 22:44:15 +08:00
protected DrawableRuleset DrawableRuleset { get ; private set ; }
2018-04-13 17:19:50 +08:00
2019-02-28 19:01:15 +08:00
protected HUDOverlay HUDOverlay { get ; private set ; }
2018-04-13 17:19:50 +08:00
2019-03-19 22:44:15 +08:00
public bool LoadedBeatmapSuccessfully = > DrawableRuleset ? . Objects . Any ( ) = = true ;
2018-04-13 17:19:50 +08:00
2019-03-17 23:46:15 +08:00
protected GameplayClockContainer GameplayClockContainer { get ; private set ; }
2018-04-13 17:19:50 +08:00
2019-11-25 15:24:29 +08:00
public DimmableStoryboard DimmableStoryboard { get ; private set ; }
2018-04-13 17:19:50 +08:00
2019-04-09 12:33:16 +08:00
[Cached]
2019-04-17 15:11:59 +08:00
[Cached(Type = typeof(IBindable<IReadOnlyList<Mod>>))]
2019-04-25 16:36:17 +08:00
protected new readonly Bindable < IReadOnlyList < Mod > > Mods = new Bindable < IReadOnlyList < Mod > > ( Array . Empty < Mod > ( ) ) ;
2019-04-09 12:33:16 +08:00
2019-09-19 03:49:28 +08:00
/// <summary>
2019-09-19 12:58:54 +08:00
/// Whether failing should be allowed.
2019-09-19 13:31:11 +08:00
/// By default, this checks whether all selected mods allow failing.
2019-09-19 03:49:28 +08:00
/// </summary>
2020-05-12 19:08:35 +08:00
protected virtual bool CheckModsAllowFailure ( ) = > Mods . Value . OfType < IApplicableFailOverride > ( ) . All ( m = > m . PerformFail ( ) ) ;
2019-09-19 03:49:28 +08:00
2020-12-23 16:39:08 +08:00
public readonly PlayerConfiguration Configuration ;
2019-03-26 15:53:44 +08:00
/// <summary>
/// Create a new player instance.
/// </summary>
2020-12-23 16:39:08 +08:00
public Player ( PlayerConfiguration configuration = null )
2019-03-26 15:53:44 +08:00
{
2020-12-23 17:07:38 +08:00
Configuration = configuration ? ? new PlayerConfiguration ( ) ;
2019-03-26 15:53:44 +08:00
}
2018-04-13 17:19:50 +08:00
2020-02-14 11:30:11 +08:00
private GameplayBeatmap gameplayBeatmap ;
2020-06-18 22:35:03 +08:00
private ScreenSuspensionHandler screenSuspension ;
2020-02-14 11:30:11 +08:00
private DependencyContainer dependencies ;
protected override IReadOnlyDependencyContainer CreateChildDependencies ( IReadOnlyDependencyContainer parent )
= > dependencies = new DependencyContainer ( base . CreateChildDependencies ( parent ) ) ;
2020-03-23 18:31:43 +08:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2020-10-23 13:47:21 +08:00
// replays should never be recorded or played back when autoplay is enabled
if ( ! Mods . Value . Any ( m = > m is ModAutoplay ) )
PrepareReplay ( ) ;
2020-03-23 18:31:43 +08:00
}
2020-12-17 15:17:13 +08:00
[CanBeNull]
2020-12-14 15:52:14 +08:00
private Score recordingScore ;
2020-03-23 18:31:43 +08:00
/// <summary>
/// Run any recording / playback setup for replays.
/// </summary>
protected virtual void PrepareReplay ( )
{
2020-12-17 15:14:41 +08:00
DrawableRuleset . SetRecordTarget ( recordingScore = new Score ( ) ) ;
2020-12-14 16:33:33 +08:00
ScoreProcessor . NewJudgement + = result = > ScoreProcessor . PopulateScore ( recordingScore . ScoreInfo ) ;
2020-03-23 18:31:43 +08:00
}
2020-10-07 13:46:58 +08:00
[BackgroundDependencyLoader(true)]
private void load ( AudioManager audio , OsuConfigManager config , OsuGame game )
2018-04-13 17:19:50 +08:00
{
2019-04-17 15:11:59 +08:00
Mods . Value = base . Mods . Value . Select ( m = > m . CreateCopy ( ) ) . ToArray ( ) ;
2019-04-09 12:33:16 +08:00
2019-12-12 14:58:11 +08:00
if ( Beatmap . Value is DummyWorkingBeatmap )
return ;
IBeatmap playableBeatmap = loadPlayableBeatmap ( ) ;
2019-03-06 19:30:14 +08:00
2019-12-12 14:58:11 +08:00
if ( playableBeatmap = = null )
2018-04-13 17:19:50 +08:00
return ;
2019-05-28 16:06:01 +08:00
sampleRestart = audio . Samples . Get ( @"Gameplay/restart" ) ;
2018-04-13 17:19:50 +08:00
mouseWheelDisabled = config . GetBindable < bool > ( OsuSetting . MouseDisableWheel ) ;
2020-10-07 13:46:58 +08:00
if ( game ! = null )
2020-10-07 15:22:39 +08:00
LocalUserPlaying . BindTo ( game . LocalUserPlaying ) ;
2020-10-06 20:09:35 +08:00
2019-12-12 14:58:11 +08:00
DrawableRuleset = ruleset . CreateDrawableRulesetWith ( playableBeatmap , Mods . Value ) ;
2019-12-24 16:01:17 +08:00
ScoreProcessor = ruleset . CreateScoreProcessor ( ) ;
ScoreProcessor . ApplyBeatmap ( playableBeatmap ) ;
2019-04-25 18:56:57 +08:00
ScoreProcessor . Mods . BindTo ( Mods ) ;
2018-04-13 17:19:50 +08:00
2019-12-27 15:14:49 +08:00
HealthProcessor = ruleset . CreateHealthProcessor ( playableBeatmap . HitObjects [ 0 ] . StartTime ) ;
2019-12-24 16:01:17 +08:00
HealthProcessor . ApplyBeatmap ( playableBeatmap ) ;
2019-12-19 19:03:14 +08:00
2018-06-29 15:49:11 +08:00
if ( ! ScoreProcessor . Mode . Disabled )
config . BindWith ( OsuSetting . ScoreDisplayMode , ScoreProcessor . Mode ) ;
2018-04-13 17:19:50 +08:00
2020-10-27 17:56:28 +08:00
InternalChild = GameplayClockContainer = CreateGameplayClockContainer ( Beatmap . Value , DrawableRuleset . GameplayStartTime ) ;
2018-04-13 17:19:50 +08:00
2020-02-14 11:30:11 +08:00
AddInternal ( gameplayBeatmap = new GameplayBeatmap ( playableBeatmap ) ) ;
2020-06-18 22:35:03 +08:00
AddInternal ( screenSuspension = new ScreenSuspensionHandler ( GameplayClockContainer ) ) ;
2020-02-14 11:30:11 +08:00
dependencies . CacheAs ( gameplayBeatmap ) ;
2020-09-29 13:09:51 +08:00
var beatmapSkinProvider = new BeatmapSkinProvidingContainer ( Beatmap . Value . Skin ) ;
// the beatmapSkinProvider is used as the fallback source here to allow the ruleset-specific skin implementation
// full access to all skin sources.
var rulesetSkinProvider = new SkinProvidingContainer ( ruleset . CreateLegacySkinProvider ( beatmapSkinProvider , playableBeatmap ) ) ;
// load the skinning hierarchy first.
// this is intentionally done in two stages to ensure things are in a loaded state before exposing the ruleset to skin sources.
GameplayClockContainer . Add ( beatmapSkinProvider . WithChild ( rulesetSkinProvider ) ) ;
rulesetSkinProvider . AddRange ( new [ ]
{
// underlay and gameplay should have access the to skinning sources.
createUnderlayComponents ( ) ,
createGameplayComponents ( Beatmap . Value , playableBeatmap )
} ) ;
2020-10-16 17:19:09 +08:00
// also give the HUD a ruleset container to allow rulesets to potentially override HUD elements (used to disable combo counters etc.)
// we may want to limit this in the future to disallow rulesets from outright replacing elements the user expects to be there.
var hudRulesetContainer = new SkinProvidingContainer ( ruleset . CreateLegacySkinProvider ( beatmapSkinProvider , playableBeatmap ) ) ;
2020-09-29 13:09:51 +08:00
// add the overlay components as a separate step as they proxy some elements from the above underlay/gameplay components.
2020-10-16 17:19:09 +08:00
GameplayClockContainer . Add ( hudRulesetContainer . WithChild ( createOverlayComponents ( Beatmap . Value ) ) ) ;
2018-04-13 17:19:50 +08:00
2020-05-08 15:37:50 +08:00
if ( ! DrawableRuleset . AllowGameplayOverlays )
2020-05-07 14:52:36 +08:00
{
HUDOverlay . ShowHud . Value = false ;
HUDOverlay . ShowHud . Disabled = true ;
BreakOverlay . Hide ( ) ;
2020-05-13 03:12:48 +08:00
skipOverlay . Hide ( ) ;
2020-05-07 14:52:36 +08:00
}
2020-10-27 17:13:45 +08:00
DrawableRuleset . FrameStableClock . WaitingOnFrames . BindValueChanged ( waiting = >
{
if ( waiting . NewValue )
GameplayClockContainer . Stop ( ) ;
else
GameplayClockContainer . Start ( ) ;
} ) ;
2020-10-27 13:10:12 +08:00
DrawableRuleset . IsPaused . BindValueChanged ( paused = >
{
updateGameplayState ( ) ;
updateSampleDisabledState ( ) ;
} ) ;
DrawableRuleset . FrameStableClock . IsCatchingUp . BindValueChanged ( _ = > updateSampleDisabledState ( ) ) ;
2020-10-27 12:54:33 +08:00
2020-10-06 20:09:35 +08:00
DrawableRuleset . HasReplayLoaded . BindValueChanged ( _ = > updateGameplayState ( ) ) ;
2020-08-16 23:18:40 +08:00
2020-03-06 17:00:17 +08:00
DrawableRuleset . HasReplayLoaded . BindValueChanged ( _ = > updatePauseOnFocusLostState ( ) , true ) ;
2018-04-13 17:19:50 +08:00
2019-08-27 17:27:21 +08:00
// bind clock into components that require it
DrawableRuleset . IsPaused . BindTo ( GameplayClockContainer . IsPaused ) ;
2018-04-13 17:19:50 +08:00
2020-11-10 22:32:30 +08:00
DrawableRuleset . NewResult + = r = >
2019-12-19 19:03:14 +08:00
{
HealthProcessor . ApplyResult ( r ) ;
ScoreProcessor . ApplyResult ( r ) ;
2020-05-03 22:55:44 +08:00
gameplayBeatmap . ApplyResult ( r ) ;
2019-12-19 19:03:14 +08:00
} ;
2020-11-10 22:32:30 +08:00
DrawableRuleset . RevertResult + = r = >
2019-12-19 19:03:14 +08:00
{
HealthProcessor . RevertResult ( r ) ;
ScoreProcessor . RevertResult ( r ) ;
} ;
2019-12-11 16:25:06 +08:00
2019-12-19 19:03:14 +08:00
// Bind the judgement processors to ourselves
2020-04-19 10:58:22 +08:00
ScoreProcessor . HasCompleted . ValueChanged + = updateCompletionState ;
2019-12-19 19:03:14 +08:00
HealthProcessor . Failed + = onFail ;
2018-04-13 17:19:50 +08:00
2019-08-27 17:27:21 +08:00
foreach ( var mod in Mods . Value . OfType < IApplicableToScoreProcessor > ( ) )
mod . ApplyToScoreProcessor ( ScoreProcessor ) ;
2019-12-19 19:03:14 +08:00
foreach ( var mod in Mods . Value . OfType < IApplicableToHealthProcessor > ( ) )
mod . ApplyToHealthProcessor ( HealthProcessor ) ;
2020-10-11 20:46:55 +08:00
IsBreakTime . BindTo ( breakTracker . IsBreakTime ) ;
2020-10-11 20:51:48 +08:00
IsBreakTime . BindValueChanged ( onBreakTimeChanged , true ) ;
2019-08-27 17:27:21 +08:00
}
2018-04-13 17:19:50 +08:00
2020-10-27 17:56:28 +08:00
protected virtual GameplayClockContainer CreateGameplayClockContainer ( WorkingBeatmap beatmap , double gameplayStart ) = > new GameplayClockContainer ( beatmap , gameplayStart ) ;
2020-09-29 13:09:51 +08:00
private Drawable createUnderlayComponents ( ) = >
DimmableStoryboard = new DimmableStoryboard ( Beatmap . Value . Storyboard ) { RelativeSizeAxes = Axes . Both } ;
2018-04-13 17:19:50 +08:00
2020-09-29 13:09:51 +08:00
private Drawable createGameplayComponents ( WorkingBeatmap working , IBeatmap playableBeatmap ) = > new ScalingContainer ( ScalingMode . Gameplay )
2019-08-27 17:27:21 +08:00
{
2020-09-29 13:09:51 +08:00
Children = new Drawable [ ]
2019-08-26 11:21:49 +08:00
{
2020-09-29 13:09:51 +08:00
DrawableRuleset . With ( r = >
r . FrameStableComponents . Children = new Drawable [ ]
{
ScoreProcessor ,
HealthProcessor ,
2020-11-13 12:35:01 +08:00
new ComboEffects ( ScoreProcessor ) ,
2020-09-29 13:09:51 +08:00
breakTracker = new BreakTracker ( DrawableRuleset . GameplayStartTime , ScoreProcessor )
{
Breaks = working . Beatmap . Breaks
}
} ) ,
}
} ;
2018-04-13 17:19:50 +08:00
2020-12-23 16:39:08 +08:00
private Drawable createOverlayComponents ( WorkingBeatmap working )
2019-08-27 17:27:21 +08:00
{
2020-12-23 16:39:08 +08:00
var container = new Container
2018-04-13 17:19:50 +08:00
{
2020-12-23 16:39:08 +08:00
RelativeSizeAxes = Axes . Both ,
Children = new [ ]
2020-03-28 04:19:49 +08:00
{
2020-12-23 16:39:08 +08:00
DimmableStoryboard . OverlayLayerContainer . CreateProxy ( ) ,
BreakOverlay = new BreakOverlay ( working . Beatmap . BeatmapInfo . LetterboxInBreaks , ScoreProcessor )
2018-04-13 17:19:50 +08:00
{
2020-12-23 16:39:08 +08:00
Clock = DrawableRuleset . FrameStableClock ,
ProcessCustomClock = false ,
Breaks = working . Beatmap . Breaks
2019-05-10 14:51:12 +08:00
} ,
2020-12-23 16:39:08 +08:00
// display the cursor above some HUD elements.
DrawableRuleset . Cursor ? . CreateProxy ( ) ? ? new Container ( ) ,
DrawableRuleset . ResumeOverlay ? . CreateProxy ( ) ? ? new Container ( ) ,
HUDOverlay = new HUDOverlay ( ScoreProcessor , HealthProcessor , DrawableRuleset , Mods . Value )
2020-03-06 17:00:17 +08:00
{
2020-12-23 16:39:08 +08:00
HoldToQuit =
{
Action = performUserRequestedExit ,
IsPaused = { BindTarget = GameplayClockContainer . IsPaused }
} ,
PlayerSettingsOverlay = { PlaybackSettings = { UserPlaybackRate = { BindTarget = GameplayClockContainer . UserPlaybackRate } } } ,
KeyCounter =
{
AlwaysVisible = { BindTarget = DrawableRuleset . HasReplayLoaded } ,
IsCounting = false
} ,
RequestSeek = time = >
{
GameplayClockContainer . Seek ( time ) ;
GameplayClockContainer . Start ( ) ;
} ,
Anchor = Anchor . Centre ,
Origin = Anchor . Centre
2020-03-06 17:00:17 +08:00
} ,
2020-12-23 16:39:08 +08:00
skipOverlay = new SkipOverlay ( DrawableRuleset . GameplayStartTime )
2020-11-24 14:41:56 +08:00
{
2020-12-23 16:39:08 +08:00
RequestSkip = GameplayClockContainer . Skip
2020-11-24 14:41:56 +08:00
} ,
2020-12-23 16:39:08 +08:00
FailOverlay = new FailOverlay
2018-04-13 17:19:50 +08:00
{
2020-12-23 16:39:08 +08:00
OnRetry = Restart ,
OnQuit = performUserRequestedExit ,
} ,
PauseOverlay = new PauseOverlay
{
OnResume = Resume ,
Retries = RestartCount ,
OnRetry = Restart ,
OnQuit = performUserRequestedExit ,
} ,
new HotkeyExitOverlay
{
Action = ( ) = >
{
if ( ! this . IsCurrentScreen ( ) ) return ;
2018-04-13 17:19:50 +08:00
2020-12-23 16:39:08 +08:00
fadeOut ( true ) ;
2020-12-24 12:31:54 +08:00
PerformExit ( true ) ;
2020-12-23 16:39:08 +08:00
} ,
2018-04-13 17:19:50 +08:00
} ,
2020-12-23 16:39:08 +08:00
failAnimation = new FailAnimation ( DrawableRuleset ) { OnComplete = onFailComplete , } ,
}
} ;
if ( Configuration . AllowRestart )
{
container . Add ( new HotkeyRetryOverlay
2019-06-24 17:15:27 +08:00
{
Action = ( ) = >
{
if ( ! this . IsCurrentScreen ( ) ) return ;
fadeOut ( true ) ;
2020-12-23 16:39:08 +08:00
Restart ( ) ;
2019-06-24 17:15:27 +08:00
} ,
2020-12-23 16:39:08 +08:00
} ) ;
2020-09-29 13:09:51 +08:00
}
2020-12-23 16:39:08 +08:00
return container ;
}
2018-04-13 17:19:50 +08:00
2020-03-06 17:00:17 +08:00
private void onBreakTimeChanged ( ValueChangedEvent < bool > isBreakTime )
2020-02-29 23:37:42 +08:00
{
2020-10-11 20:51:48 +08:00
updateGameplayState ( ) ;
2020-03-06 17:00:17 +08:00
updatePauseOnFocusLostState ( ) ;
HUDOverlay . KeyCounter . IsCounting = ! isBreakTime . NewValue ;
2020-02-29 23:37:42 +08:00
}
2020-10-06 20:09:35 +08:00
private void updateGameplayState ( )
2020-08-04 03:25:45 +08:00
{
2020-10-06 20:09:35 +08:00
bool inGameplay = ! DrawableRuleset . HasReplayLoaded . Value & & ! DrawableRuleset . IsPaused . Value & & ! breakTracker . IsBreakTime . Value ;
OverlayActivationMode . Value = inGameplay ? OverlayActivation . Disabled : OverlayActivation . UserTriggered ;
2020-10-07 15:22:39 +08:00
LocalUserPlaying . Value = inGameplay ;
2020-08-04 03:25:45 +08:00
}
2020-10-27 13:10:12 +08:00
private void updateSampleDisabledState ( )
{
samplePlaybackDisabled . Value = DrawableRuleset . FrameStableClock . IsCatchingUp . Value | | GameplayClockContainer . GameplayClock . IsPaused . Value ;
}
2020-03-06 17:00:17 +08:00
private void updatePauseOnFocusLostState ( ) = >
2019-12-11 14:45:50 +08:00
HUDOverlay . HoldToQuit . PauseOnFocusLost = PauseOnFocusLost
2020-03-06 17:00:17 +08:00
& & ! DrawableRuleset . HasReplayLoaded . Value
2020-03-26 14:28:56 +08:00
& & ! breakTracker . IsBreakTime . Value ;
2019-12-11 14:45:50 +08:00
2019-12-12 14:58:11 +08:00
private IBeatmap loadPlayableBeatmap ( )
2018-04-13 17:19:50 +08:00
{
2019-12-12 14:58:11 +08:00
IBeatmap playable ;
2019-03-06 19:30:14 +08:00
try
{
2019-12-12 14:58:11 +08:00
if ( Beatmap . Value . Beatmap = = null )
2019-03-06 19:30:14 +08:00
throw new InvalidOperationException ( "Beatmap was not loaded" ) ;
2018-04-13 17:19:50 +08:00
2019-12-12 14:58:11 +08:00
rulesetInfo = Ruleset . Value ? ? Beatmap . Value . BeatmapInfo . Ruleset ;
2019-08-26 11:21:49 +08:00
ruleset = rulesetInfo . CreateInstance ( ) ;
2019-03-06 19:30:14 +08:00
try
{
2019-12-12 14:58:11 +08:00
playable = Beatmap . Value . GetPlayableBeatmap ( ruleset . RulesetInfo , Mods . Value ) ;
2019-03-06 19:30:14 +08:00
}
catch ( BeatmapInvalidForRulesetException )
{
2019-12-12 14:58:11 +08:00
// A playable beatmap may not be creatable with the user's preferred ruleset, so try using the beatmap's default ruleset
rulesetInfo = Beatmap . Value . BeatmapInfo . Ruleset ;
2019-08-26 11:21:49 +08:00
ruleset = rulesetInfo . CreateInstance ( ) ;
2019-12-12 14:58:11 +08:00
playable = Beatmap . Value . GetPlayableBeatmap ( rulesetInfo , Mods . Value ) ;
2018-04-13 17:19:50 +08:00
}
2019-12-12 14:58:11 +08:00
if ( playable . HitObjects . Count = = 0 )
2019-03-06 19:30:14 +08:00
{
Logger . Log ( "Beatmap contains no hit objects!" , level : LogLevel . Error ) ;
return null ;
}
}
catch ( Exception e )
{
2020-09-23 15:30:20 +08:00
Logger . Error ( e , "Could not load beatmap successfully!" ) ;
2019-03-06 19:30:14 +08:00
//couldn't load, hard abort!
return null ;
}
2018-04-22 01:21:09 +08:00
2019-12-12 14:58:11 +08:00
return playable ;
2018-04-13 17:19:50 +08:00
}
2018-07-19 00:18:07 +08:00
2020-12-23 22:51:26 +08:00
/// <summary>
/// Exits the <see cref="Player"/>.
/// </summary>
/// <param name="userRequested">
/// Whether the exit is requested by the user, or a higher-level game component.
/// Pausing is allowed only in the former case.
/// </param>
protected void PerformExit ( bool userRequested )
2018-12-13 15:17:24 +08:00
{
2019-06-24 17:15:27 +08:00
// if a restart has been requested, cancel any pending completion (user has shown intent to restart).
2019-08-06 22:05:12 +08:00
completionProgressDelegate ? . Cancel ( ) ;
2018-04-13 17:19:50 +08:00
2019-06-24 17:15:27 +08:00
ValidForResume = false ;
2018-04-13 17:19:50 +08:00
2020-12-23 22:51:26 +08:00
if ( ! this . IsCurrentScreen ( ) ) return ;
if ( userRequested )
performUserRequestedExit ( ) ;
else
this . Exit ( ) ;
2018-04-13 17:19:50 +08:00
}
2019-06-25 22:15:58 +08:00
private void performUserRequestedExit ( )
2018-04-13 17:19:50 +08:00
{
2019-10-04 11:41:53 +08:00
if ( ValidForResume & & HasFailed & & ! FailOverlay . IsPresent )
2019-10-04 11:23:42 +08:00
{
failAnimation . FinishTransforms ( true ) ;
return ;
}
2019-10-03 02:16:31 +08:00
if ( canPause )
Pause ( ) ;
else
this . Exit ( ) ;
2018-04-13 17:19:50 +08:00
}
2019-11-01 14:51:45 +08:00
/// <summary>
/// Restart gameplay via a parent <see cref="PlayerLoader"/>.
/// <remarks>This can be called from a child screen in order to trigger the restart process.</remarks>
/// </summary>
2018-04-13 17:19:50 +08:00
public void Restart ( )
{
2020-12-23 16:39:08 +08:00
if ( ! Configuration . AllowRestart )
return ;
2020-10-07 16:40:54 +08:00
// at the point of restarting the track should either already be paused or the volume should be zero.
// stopping here is to ensure music doesn't become audible after exiting back to PlayerLoader.
musicController . Stop ( ) ;
2018-04-13 17:19:50 +08:00
sampleRestart ? . Play ( ) ;
RestartRequested ? . Invoke ( ) ;
2019-11-01 14:32:06 +08:00
if ( this . IsCurrentScreen ( ) )
2020-12-23 22:51:26 +08:00
PerformExit ( true ) ;
2019-11-01 14:51:10 +08:00
else
this . MakeCurrent ( ) ;
2018-04-13 17:19:50 +08:00
}
2019-08-06 22:05:12 +08:00
private ScheduledDelegate completionProgressDelegate ;
2020-12-18 16:48:42 +08:00
private Task < ScoreInfo > scoreSubmissionTask ;
2018-04-13 17:19:50 +08:00
2020-04-19 10:58:22 +08:00
private void updateCompletionState ( ValueChangedEvent < bool > completionState )
2018-04-13 17:19:50 +08:00
{
2020-03-19 13:10:54 +08:00
// screen may be in the exiting transition phase.
if ( ! this . IsCurrentScreen ( ) )
return ;
2020-04-19 10:59:56 +08:00
if ( ! completionState . NewValue )
{
completionProgressDelegate ? . Cancel ( ) ;
completionProgressDelegate = null ;
ValidForResume = true ;
return ;
}
2020-04-21 10:51:20 +08:00
if ( completionProgressDelegate ! = null )
throw new InvalidOperationException ( $"{nameof(updateCompletionState)} was fired more than once" ) ;
2018-04-13 17:19:50 +08:00
// Only show the completion screen if the player hasn't failed
2020-04-21 10:51:20 +08:00
if ( HealthProcessor . HasFailed )
2018-04-13 17:19:50 +08:00
return ;
ValidForResume = false ;
2020-12-23 16:39:08 +08:00
if ( ! Configuration . ShowResults ) return ;
2018-04-13 17:19:50 +08:00
2020-12-18 17:20:36 +08:00
scoreSubmissionTask ? ? = Task . Run ( async ( ) = >
2020-12-18 15:51:59 +08:00
{
2020-12-18 17:20:36 +08:00
var score = CreateScore ( ) ;
try
2020-12-18 15:51:59 +08:00
{
2020-12-19 02:32:05 +08:00
await SubmitScore ( score ) ;
2020-12-18 17:20:36 +08:00
}
catch ( Exception ex )
{
Logger . Error ( ex , "Score submission failed!" ) ;
2020-12-18 15:51:59 +08:00
}
2020-12-19 02:32:05 +08:00
try
{
await ImportScore ( score ) ;
}
catch ( Exception ex )
{
Logger . Error ( ex , "Score import failed!" ) ;
}
return score . ScoreInfo ;
2020-12-18 17:20:36 +08:00
} ) ;
using ( BeginDelayedSequence ( RESULTS_DISPLAY_DELAY ) )
scheduleCompletion ( ) ;
2018-04-13 17:19:50 +08:00
}
2020-12-18 17:20:36 +08:00
private void scheduleCompletion ( ) = > completionProgressDelegate = Schedule ( ( ) = >
{
if ( ! scoreSubmissionTask . IsCompleted )
{
scheduleCompletion ( ) ;
return ;
}
// screen may be in the exiting transition phase.
if ( this . IsCurrentScreen ( ) )
this . Push ( CreateResults ( scoreSubmissionTask . Result ) ) ;
} ) ;
2019-03-18 10:48:11 +08:00
protected override bool OnScroll ( ScrollEvent e ) = > mouseWheelDisabled . Value & & ! GameplayClockContainer . IsPaused . Value ;
#region Fail Logic
protected FailOverlay FailOverlay { get ; private set ; }
2019-06-04 15:13:16 +08:00
private FailAnimation failAnimation ;
2018-04-13 17:19:50 +08:00
private bool onFail ( )
{
2020-05-12 19:08:35 +08:00
if ( ! CheckModsAllowFailure ( ) )
2018-04-13 17:19:50 +08:00
return false ;
2018-10-31 19:03:37 +08:00
HasFailed = true ;
2019-03-18 10:48:11 +08:00
// There is a chance that we could be in a paused state as the ruleset's internal clock (see FrameStabilityContainer)
// could process an extra frame after the GameplayClock is stopped.
// In such cases we want the fail state to precede a user triggered pause.
2019-06-11 13:28:52 +08:00
if ( PauseOverlay . State . Value = = Visibility . Visible )
2019-03-18 10:48:11 +08:00
PauseOverlay . Hide ( ) ;
2019-09-19 00:45:59 +08:00
failAnimation . Start ( ) ;
2019-09-19 16:53:10 +08:00
if ( Mods . Value . OfType < IApplicableFailOverride > ( ) . Any ( m = > m . RestartOnFail ) )
2018-10-14 23:18:52 +08:00
Restart ( ) ;
2018-04-13 17:19:50 +08:00
return true ;
}
2019-06-04 15:13:16 +08:00
// Called back when the transform finishes
private void onFailComplete ( )
{
GameplayClockContainer . Stop ( ) ;
2019-03-18 10:48:11 +08:00
FailOverlay . Retries = RestartCount ;
FailOverlay . Show ( ) ;
2018-04-13 17:19:50 +08:00
}
2019-03-18 10:48:11 +08:00
#endregion
#region Pause Logic
public bool IsResuming { get ; private set ; }
/// <summary>
/// The amount of gameplay time after which a second pause is allowed.
/// </summary>
private const double pause_cooldown = 1000 ;
protected PauseOverlay PauseOverlay { get ; private set ; }
private double? lastPauseActionTime ;
private bool canPause = >
// must pass basic screen conditions (beatmap loaded, instance allows pause)
2020-12-23 16:39:08 +08:00
LoadedBeatmapSuccessfully & & Configuration . AllowPause & & ValidForResume
2019-03-18 10:48:11 +08:00
// replays cannot be paused and exit immediately
2019-03-20 14:27:06 +08:00
& & ! DrawableRuleset . HasReplayLoaded . Value
2019-03-18 10:48:11 +08:00
// cannot pause if we are already in a fail state
& & ! HasFailed
2019-03-22 13:42:51 +08:00
// cannot pause if already paused (or in a cooldown state) unless we are in a resuming state.
2019-11-24 15:44:35 +08:00
& & ( IsResuming | | ( GameplayClockContainer . IsPaused . Value = = false & & ! pauseCooldownActive ) ) ;
2019-03-18 13:40:53 +08:00
private bool pauseCooldownActive = >
lastPauseActionTime . HasValue & & GameplayClockContainer . GameplayClock . CurrentTime < lastPauseActionTime + pause_cooldown ;
2019-03-18 10:48:11 +08:00
private bool canResume = >
// cannot resume from a non-paused state
GameplayClockContainer . IsPaused . Value
// cannot resume if we are already in a fail state
& & ! HasFailed
// already resuming
& & ! IsResuming ;
public void Pause ( )
{
if ( ! canPause ) return ;
2019-10-26 03:57:49 +08:00
if ( IsResuming )
{
DrawableRuleset . CancelResume ( ) ;
2019-11-01 13:43:52 +08:00
IsResuming = false ;
2019-10-26 03:57:49 +08:00
}
2019-03-18 10:48:11 +08:00
GameplayClockContainer . Stop ( ) ;
PauseOverlay . Show ( ) ;
lastPauseActionTime = GameplayClockContainer . GameplayClock . CurrentTime ;
}
public void Resume ( )
{
if ( ! canResume ) return ;
IsResuming = true ;
PauseOverlay . Hide ( ) ;
2019-03-18 13:40:53 +08:00
2019-04-08 03:32:55 +08:00
// breaks and time-based conditions may allow instant resume.
2020-03-26 14:28:56 +08:00
if ( breakTracker . IsBreakTime . Value )
2019-03-18 13:40:53 +08:00
completeResume ( ) ;
else
2019-03-20 14:27:06 +08:00
DrawableRuleset . RequestResume ( completeResume ) ;
2019-03-18 13:40:53 +08:00
void completeResume ( )
{
GameplayClockContainer . Start ( ) ;
IsResuming = false ;
}
2019-03-18 10:48:11 +08:00
}
#endregion
#region Screen Logic
2019-01-23 19:52:00 +08:00
public override void OnEntering ( IScreen last )
2018-04-13 17:19:50 +08:00
{
base . OnEntering ( last ) ;
2018-04-20 16:30:27 +08:00
if ( ! LoadedBeatmapSuccessfully )
2018-04-13 17:19:50 +08:00
return ;
2019-01-23 19:52:00 +08:00
Alpha = 0 ;
this
2018-04-13 17:19:50 +08:00
. ScaleTo ( 0.7f )
. ScaleTo ( 1 , 750 , Easing . OutQuint )
. Delay ( 250 )
. FadeIn ( 250 ) ;
2019-11-25 15:24:29 +08:00
Background . EnableUserDim . Value = true ;
Background . BlurAmount . Value = 0 ;
2018-04-13 17:19:50 +08:00
2019-12-11 04:06:13 +08:00
// bind component bindables.
2020-03-26 14:28:56 +08:00
Background . IsBreakTime . BindTo ( breakTracker . IsBreakTime ) ;
2020-07-22 11:41:06 +08:00
HUDOverlay . IsBreakTime . BindTo ( breakTracker . IsBreakTime ) ;
2020-03-26 14:28:56 +08:00
DimmableStoryboard . IsBreakTime . BindTo ( breakTracker . IsBreakTime ) ;
2019-12-11 04:06:13 +08:00
2019-02-28 19:01:15 +08:00
Background . StoryboardReplacesBackground . BindTo ( storyboardReplacesBackground ) ;
2019-07-12 10:50:06 +08:00
DimmableStoryboard . StoryboardReplacesBackground . BindTo ( storyboardReplacesBackground ) ;
2018-04-13 17:19:50 +08:00
2019-02-25 12:15:37 +08:00
storyboardReplacesBackground . Value = Beatmap . Value . Storyboard . ReplacesBackground & & Beatmap . Value . Storyboard . HasDrawable ;
2019-02-18 15:34:11 +08:00
2019-11-25 15:24:29 +08:00
foreach ( var mod in Mods . Value . OfType < IApplicableToPlayer > ( ) )
mod . ApplyToPlayer ( this ) ;
2019-06-29 09:23:59 +08:00
foreach ( var mod in Mods . Value . OfType < IApplicableToHUD > ( ) )
mod . ApplyToHUD ( HUDOverlay ) ;
2020-08-03 03:34:35 +08:00
2020-09-01 17:07:19 +08:00
// Our mods are local copies of the global mods so they need to be re-applied to the track.
// This is done through the music controller (for now), because resetting speed adjustments on the beatmap track also removes adjustments provided by DrawableTrack.
// Todo: In the future, player will receive in a track and will probably not have to worry about this...
musicController . ResetTrackAdjustments ( ) ;
2020-09-01 15:55:10 +08:00
foreach ( var mod in Mods . Value . OfType < IApplicableToTrack > ( ) )
2020-09-01 17:07:19 +08:00
mod . ApplyToTrack ( musicController . CurrentTrack ) ;
2020-09-04 03:56:47 +08:00
2020-10-06 20:09:35 +08:00
updateGameplayState ( ) ;
2020-12-24 14:32:55 +08:00
GameplayClockContainer . FadeInFromZero ( 750 , Easing . OutQuint ) ;
StartGameplay ( ) ;
}
/// <summary>
/// Called to trigger the starting of the gameplay clock and underlying gameplay.
/// This will be called on entering the player screen once. A derived class may block the first call to this to delay the start of gameplay.
/// </summary>
protected virtual void StartGameplay ( )
{
if ( GameplayClockContainer . GameplayClock . IsRunning )
throw new InvalidOperationException ( $"{nameof(StartGameplay)} should not be called when the gameplay clock is already running" ) ;
GameplayClockContainer . Restart ( ) ;
2018-04-13 17:19:50 +08:00
}
2019-01-23 19:52:00 +08:00
public override void OnSuspending ( IScreen next )
2018-04-13 17:19:50 +08:00
{
2020-06-18 22:35:03 +08:00
screenSuspension ? . Expire ( ) ;
2018-04-13 17:19:50 +08:00
fadeOut ( ) ;
base . OnSuspending ( next ) ;
}
2019-01-23 19:52:00 +08:00
public override bool OnExiting ( IScreen next )
2018-04-13 17:19:50 +08:00
{
2020-06-18 22:35:03 +08:00
screenSuspension ? . Expire ( ) ;
2019-08-06 22:05:12 +08:00
if ( completionProgressDelegate ! = null & & ! completionProgressDelegate . Cancelled & & ! completionProgressDelegate . Completed )
2018-07-28 06:34:51 +08:00
{
2019-08-06 22:05:12 +08:00
// proceed to result screen if beatmap already finished playing
2020-03-19 13:10:54 +08:00
completionProgressDelegate . RunTask ( ) ;
2018-07-28 06:34:51 +08:00
return true ;
}
2019-09-13 14:41:53 +08:00
// ValidForResume is false when restarting
if ( ValidForResume )
2019-06-04 15:13:16 +08:00
{
2019-09-13 14:41:53 +08:00
if ( pauseCooldownActive & & ! GameplayClockContainer . IsPaused . Value )
// still want to block if we are within the cooldown period and not already paused.
return true ;
2018-04-13 17:19:50 +08:00
}
2019-11-01 13:11:18 +08:00
// GameplayClockContainer performs seeks / start / stop operations on the beatmap's track.
// as we are no longer the current screen, we cannot guarantee the track is still usable.
2019-12-23 18:13:36 +08:00
GameplayClockContainer ? . StopUsingBeatmapClock ( ) ;
2019-11-01 13:11:18 +08:00
2020-09-01 17:07:19 +08:00
musicController . ResetTrackAdjustments ( ) ;
2020-09-01 15:55:10 +08:00
2019-03-16 13:20:10 +08:00
fadeOut ( ) ;
return base . OnExiting ( next ) ;
2018-04-13 17:19:50 +08:00
}
2020-12-18 16:47:33 +08:00
/// <summary>
/// Creates the player's <see cref="Score"/>.
/// </summary>
/// <returns>The <see cref="Score"/>.</returns>
2020-12-18 15:51:59 +08:00
protected virtual Score CreateScore ( )
2020-12-18 14:36:24 +08:00
{
2020-12-18 15:51:59 +08:00
var score = new Score
2020-12-18 14:36:24 +08:00
{
2020-12-18 15:51:59 +08:00
ScoreInfo = new ScoreInfo
{
Beatmap = Beatmap . Value . BeatmapInfo ,
Ruleset = rulesetInfo ,
Mods = Mods . Value . ToArray ( ) ,
}
2020-12-18 14:36:24 +08:00
} ;
if ( DrawableRuleset . ReplayScore ! = null )
2020-12-18 15:51:59 +08:00
{
score . ScoreInfo . User = DrawableRuleset . ReplayScore . ScoreInfo ? . User ? ? new GuestUser ( ) ;
score . Replay = DrawableRuleset . ReplayScore . Replay ;
}
2020-12-18 14:36:24 +08:00
else
2020-12-18 15:51:59 +08:00
{
score . ScoreInfo . User = api . LocalUser . Value ;
2020-12-18 17:31:49 +08:00
score . Replay = new Replay { Frames = recordingScore ? . Replay . Frames . ToList ( ) ? ? new List < ReplayFrame > ( ) } ;
2020-12-18 15:51:59 +08:00
}
2020-12-18 14:36:24 +08:00
2020-12-18 15:51:59 +08:00
ScoreProcessor . PopulateScore ( score . ScoreInfo ) ;
2020-12-18 14:36:24 +08:00
return score ;
}
2020-12-18 16:47:33 +08:00
/// <summary>
2020-12-19 02:32:05 +08:00
/// Imports the player's <see cref="Score"/> to the local database.
2020-12-18 16:47:33 +08:00
/// </summary>
2020-12-19 02:32:05 +08:00
/// <param name="score">The <see cref="Score"/> to import.</param>
/// <returns>The imported score.</returns>
2020-12-19 12:58:56 +08:00
protected virtual Task ImportScore ( Score score )
2020-03-29 21:51:28 +08:00
{
2020-12-18 15:51:59 +08:00
// Replays are already populated and present in the game's database, so should not be re-imported.
2020-03-29 21:51:28 +08:00
if ( DrawableRuleset . ReplayScore ! = null )
2020-12-19 12:58:56 +08:00
return Task . CompletedTask ;
2020-03-29 21:51:28 +08:00
2020-12-18 15:51:59 +08:00
LegacyByteArrayReader replayReader ;
2020-03-29 21:51:28 +08:00
2020-12-18 15:51:59 +08:00
using ( var stream = new MemoryStream ( ) )
2020-03-29 21:51:28 +08:00
{
2020-12-18 15:51:59 +08:00
new LegacyScoreEncoder ( score , gameplayBeatmap . PlayableBeatmap ) . Encode ( stream ) ;
replayReader = new LegacyByteArrayReader ( stream . ToArray ( ) , "replay.osr" ) ;
2020-03-29 21:51:28 +08:00
}
2020-12-19 12:58:56 +08:00
return scoreManager . Import ( score . ScoreInfo , replayReader ) ;
2020-03-29 21:51:28 +08:00
}
2020-12-19 02:32:05 +08:00
/// <summary>
/// Submits the player's <see cref="Score"/>.
/// </summary>
/// <param name="score">The <see cref="Score"/> to submit.</param>
/// <returns>The submitted score.</returns>
protected virtual Task SubmitScore ( Score score ) = > Task . CompletedTask ;
2020-12-18 16:47:33 +08:00
/// <summary>
/// Creates the <see cref="ResultsScreen"/> for a <see cref="ScoreInfo"/>.
/// </summary>
/// <param name="score">The <see cref="ScoreInfo"/> to be displayed in the results screen.</param>
/// <returns>The <see cref="ResultsScreen"/>.</returns>
2020-12-18 14:36:24 +08:00
protected virtual ResultsScreen CreateResults ( ScoreInfo score ) = > new SoloResultsScreen ( score , true ) ;
2018-08-02 18:08:23 +08:00
private void fadeOut ( bool instant = false )
2018-04-13 17:19:50 +08:00
{
2018-08-02 18:08:23 +08:00
float fadeOutDuration = instant ? 0 : 250 ;
2019-01-23 19:52:00 +08:00
this . FadeOut ( fadeOutDuration ) ;
2018-04-13 17:19:50 +08:00
2019-02-20 15:53:57 +08:00
Background . EnableUserDim . Value = false ;
2019-02-25 12:15:37 +08:00
storyboardReplacesBackground . Value = false ;
2018-04-13 17:19:50 +08:00
}
2019-03-18 10:48:11 +08:00
#endregion
2020-10-14 18:39:48 +08:00
IBindable < bool > ISamplePlaybackDisabler . SamplePlaybackDisabled = > samplePlaybackDisabled ;
2018-04-13 17:19:50 +08:00
}
}