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
2018-12-26 13:16:35 +00:00
using System ;
2020-12-10 07:56:56 +00:00
using System.Diagnostics ;
2018-03-23 06:04:40 +00:00
using System.Threading.Tasks ;
2021-11-10 08:08:51 +00:00
using ManagedBass.Fx ;
2017-02-22 12:43:29 +00:00
using osu.Framework.Allocation ;
2019-09-15 13:59:46 +00:00
using osu.Framework.Audio ;
2019-10-01 15:39:01 +00:00
using osu.Framework.Bindables ;
2017-02-22 12:43:29 +00:00
using osu.Framework.Graphics ;
using osu.Framework.Graphics.Containers ;
using osu.Framework.Graphics.Sprites ;
2020-10-19 22:22:30 +00:00
using osu.Framework.Graphics.Transforms ;
2019-03-20 10:35:40 +00:00
using osu.Framework.Input ;
2017-02-22 12:43:29 +00:00
using osu.Framework.Screens ;
2018-05-09 14:31:52 +00:00
using osu.Framework.Threading ;
2021-10-07 05:15:16 +00:00
using osu.Game.Audio.Effects ;
2019-10-01 15:39:01 +00:00
using osu.Game.Configuration ;
2017-02-22 12:43:29 +00:00
using osu.Game.Graphics ;
2019-03-22 10:01:32 +00:00
using osu.Game.Graphics.Containers ;
2019-07-05 06:32:07 +00:00
using osu.Game.Input ;
2019-09-15 13:59:46 +00:00
using osu.Game.Overlays ;
using osu.Game.Overlays.Notifications ;
2017-11-01 11:54:58 +00:00
using osu.Game.Screens.Menu ;
2018-01-26 20:29:54 +00:00
using osu.Game.Screens.Play.PlayerSettings ;
2019-04-13 11:18:44 +00:00
using osu.Game.Users ;
2021-04-08 23:34:35 +00:00
using osu.Game.Utils ;
2018-11-20 07:51:59 +00:00
using osuTK ;
using osuTK.Graphics ;
2018-04-13 09:19:50 +00:00
2017-02-22 12:43:29 +00:00
namespace osu.Game.Screens.Play
{
2018-02-15 20:57:53 +00:00
public class PlayerLoader : ScreenWithBeatmapBackground
2017-02-22 12:43:29 +00:00
{
2019-03-14 07:09:17 +00:00
protected const float BACKGROUND_BLUR = 15 ;
2019-03-13 07:47:03 +00:00
2021-11-12 11:30:40 +00:00
protected const double CONTENT_OUT_DURATION = 300 ;
protected virtual double PlayerPushDelay = > 1800 ;
2021-10-13 04:26:20 +00:00
2020-02-14 09:22:57 +00:00
public override bool HideOverlaysOnEnter = > hideOverlays ;
2018-04-13 09:19:50 +00:00
2020-02-14 09:22:57 +00:00
public override bool DisallowExternalBeatmapRulesetChanges = > true ;
2019-01-24 10:55:42 +00:00
2020-02-14 09:22:57 +00:00
// Here because IsHovered will not update unless we do so.
public override bool HandlePositionalInput = > true ;
2018-04-13 09:19:50 +00:00
2020-02-14 09:22:57 +00:00
// We show the previous screen status
2022-03-17 10:36:33 +00:00
protected override UserActivity ? InitialActivity = > null ;
2018-04-13 09:19:50 +00:00
2022-03-17 10:36:33 +00:00
protected BeatmapMetadataDisplay MetadataInfo { get ; private set ; } = null ! ;
2019-02-01 06:42:15 +00:00
2021-08-13 03:27:57 +00:00
/// <summary>
/// A fill flow containing the player settings groups, exposed for the ability to hide it from inheritors of the player loader.
/// </summary>
2022-03-17 10:36:33 +00:00
protected FillFlowContainer < PlayerSettingsGroup > PlayerSettings { get ; private set ; } = null ! ;
2021-08-13 03:27:57 +00:00
2022-03-17 10:36:33 +00:00
protected VisualSettings VisualSettings { get ; private set ; } = null ! ;
2019-04-25 05:15:07 +00:00
2022-03-17 10:36:33 +00:00
protected AudioSettings AudioSettings { get ; private set ; } = null ! ;
2022-03-01 07:14:57 +00:00
2022-03-17 10:36:33 +00:00
protected Task ? LoadTask { get ; private set ; }
2019-12-06 04:47:34 +00:00
2022-03-17 10:36:33 +00:00
protected Task ? DisposalTask { get ; private set ; }
2018-04-13 09:19:50 +00:00
2022-10-13 08:12:28 +00:00
private OsuScrollContainer settingsScroll = null ! ;
2020-02-14 09:22:57 +00:00
private bool backgroundBrightnessReduction ;
2020-10-25 11:33:35 +00:00
private readonly BindableDouble volumeAdjustment = new BindableDouble ( 1 ) ;
2022-03-17 10:36:33 +00:00
private AudioFilter lowPassFilter = null ! ;
private AudioFilter highPassFilter = null ! ;
2021-10-07 05:15:16 +00:00
2022-10-13 07:04:38 +00:00
[Cached]
private OverlayColourProvider colourProvider = new OverlayColourProvider ( OverlayColourScheme . Purple ) ;
2020-02-14 09:22:57 +00:00
protected bool BackgroundBrightnessReduction
{
set
{
if ( value = = backgroundBrightnessReduction )
return ;
backgroundBrightnessReduction = value ;
2021-01-04 09:32:23 +00:00
ApplyToBackground ( b = > b . FadeColour ( OsuColour . Gray ( backgroundBrightnessReduction ? 0.8f : 1 ) , 200 ) ) ;
2020-02-14 09:22:57 +00:00
}
}
private bool readyForPush = >
2020-12-10 07:56:56 +00:00
! playerConsumed
2020-02-14 10:02:37 +00:00
// don't push unless the player is completely loaded
2022-03-17 10:23:43 +00:00
& & CurrentPlayer ? . LoadState = = LoadState . Ready
2022-04-21 13:55:13 +00:00
// don't push unless the player is ready to start gameplay
& & ReadyForGameplay ;
protected virtual bool ReadyForGameplay = >
// not ready if the user is hovering one of the panes, unless they are idle.
( IsHovered | | idleTracker . IsIdle . Value )
// not ready if the user is dragging a slider or otherwise.
2022-03-17 10:36:33 +00:00
& & inputManager . DraggedDrawable = = null
2022-04-21 13:55:13 +00:00
// not ready if a focused overlay is visible, like settings.
2022-03-17 10:36:33 +00:00
& & inputManager . FocusedDrawable = = null ;
2020-02-14 09:22:57 +00:00
private readonly Func < Player > createPlayer ;
2022-03-17 10:23:43 +00:00
/// <summary>
/// The <see cref="Player"/> instance being loaded by this screen.
/// </summary>
2022-03-17 10:36:33 +00:00
public Player ? CurrentPlayer { get ; private set ; }
2020-02-14 09:22:57 +00:00
2020-12-10 07:56:56 +00:00
/// <summary>
2022-03-17 10:23:43 +00:00
/// Whether the current player instance has been consumed via <see cref="consumePlayer"/>.
2020-12-10 07:56:56 +00:00
/// </summary>
private bool playerConsumed ;
2022-03-17 10:36:33 +00:00
private LogoTrackingContainer content = null ! ;
2020-02-14 09:22:57 +00:00
private bool hideOverlays ;
2022-03-17 10:36:33 +00:00
private InputManager inputManager = null ! ;
2020-02-14 09:22:57 +00:00
2022-03-17 10:36:33 +00:00
private IdleTracker idleTracker = null ! ;
2019-09-15 14:32:23 +00:00
2022-03-17 10:36:33 +00:00
private ScheduledDelegate ? scheduledPushPlayer ;
2020-02-14 09:22:57 +00:00
2022-03-17 10:36:33 +00:00
private EpilepsyWarning ? epilepsyWarning ;
2020-10-19 22:08:05 +00:00
2022-08-16 04:04:56 +00:00
private bool quickRestart ;
2022-08-06 15:02:45 +00:00
2019-10-03 10:16:31 +00:00
[Resolved(CanBeNull = true)]
2022-04-18 21:18:10 +00:00
private INotificationOverlay ? notificationOverlay { get ; set ; }
2019-10-03 10:16:31 +00:00
[Resolved(CanBeNull = true)]
2022-03-17 10:36:33 +00:00
private VolumeOverlay ? volumeOverlay { get ; set ; }
2019-10-03 10:16:31 +00:00
[Resolved]
2022-03-17 10:36:33 +00:00
private AudioManager audioManager { get ; set ; } = null ! ;
2019-10-03 10:16:31 +00:00
2021-04-09 21:55:41 +00:00
[Resolved(CanBeNull = true)]
2022-03-17 10:36:33 +00:00
private BatteryInfo ? batteryInfo { get ; set ; }
2021-04-08 23:34:35 +00:00
2018-12-26 13:16:35 +00:00
public PlayerLoader ( Func < Player > createPlayer )
2017-02-22 12:43:29 +00:00
{
2018-12-26 13:16:35 +00:00
this . createPlayer = createPlayer ;
}
2018-04-13 09:19:50 +00:00
2017-02-22 12:43:29 +00:00
[BackgroundDependencyLoader]
2021-10-07 05:15:16 +00:00
private void load ( SessionStatics sessionStatics , AudioManager audio )
2017-02-22 12:43:29 +00:00
{
2019-10-01 15:39:01 +00:00
muteWarningShownOnce = sessionStatics . GetBindable < bool > ( Static . MutedAudioNotificationShownOnce ) ;
2021-04-09 21:55:41 +00:00
batteryWarningShownOnce = sessionStatics . GetBindable < bool > ( Static . LowBatteryNotificationShownOnce ) ;
2019-10-01 15:39:01 +00:00
2022-03-03 07:23:30 +00:00
const float padding = 25 ;
2021-10-13 04:26:20 +00:00
InternalChildren = new Drawable [ ]
2019-03-27 09:11:12 +00:00
{
2021-10-13 04:26:20 +00:00
( content = new LogoTrackingContainer
2018-05-09 14:31:52 +00:00
{
2019-03-27 09:11:12 +00:00
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
2021-10-13 04:26:20 +00:00
RelativeSizeAxes = Axes . Both ,
} ) . WithChildren ( new Drawable [ ]
2019-03-27 09:11:12 +00:00
{
2021-10-13 04:26:20 +00:00
MetadataInfo = new BeatmapMetadataDisplay ( Beatmap . Value , Mods , content . LogoFacade )
2019-01-24 10:55:42 +00:00
{
2021-10-13 04:26:20 +00:00
Alpha = 0 ,
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
} ,
2022-10-13 08:12:28 +00:00
} ) ,
settingsScroll = new OsuScrollContainer
{
Anchor = Anchor . TopRight ,
Origin = Anchor . TopRight ,
RelativeSizeAxes = Axes . Y ,
Width = SettingsToolboxGroup . CONTAINER_WIDTH + padding * 2 ,
Padding = new MarginPadding { Vertical = padding } ,
Masking = false ,
Child = PlayerSettings = new FillFlowContainer < PlayerSettingsGroup >
2021-10-13 04:26:20 +00:00
{
2022-10-13 08:12:28 +00:00
AutoSizeAxes = Axes . Both ,
Direction = FillDirection . Vertical ,
Spacing = new Vector2 ( 0 , 20 ) ,
Padding = new MarginPadding { Horizontal = padding } ,
Children = new PlayerSettingsGroup [ ]
2021-10-13 04:26:20 +00:00
{
2022-10-13 08:12:28 +00:00
VisualSettings = new VisualSettings ( ) ,
AudioSettings = new AudioSettings ( ) ,
new InputSettings ( )
}
2021-10-13 04:26:20 +00:00
} ,
2022-10-13 08:12:28 +00:00
} ,
idleTracker = new IdleTracker ( 750 ) ,
2021-11-10 08:08:51 +00:00
lowPassFilter = new AudioFilter ( audio . TrackMixer ) ,
highPassFilter = new AudioFilter ( audio . TrackMixer , BQFType . HighPass )
2021-10-13 04:26:20 +00:00
} ;
2020-10-19 22:08:05 +00:00
if ( Beatmap . Value . BeatmapInfo . EpilepsyWarning )
{
AddInternal ( epilepsyWarning = new EpilepsyWarning
{
Anchor = Anchor . Centre ,
Origin = Anchor . Centre ,
} ) ;
}
2017-02-22 12:43:29 +00:00
}
2018-04-13 09:19:50 +00:00
2019-10-03 10:16:31 +00:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
inputManager = GetContainingInputManager ( ) ;
2019-12-27 10:36:48 +00:00
}
2020-02-14 09:28:58 +00:00
#region Screen handling
2022-04-21 15:52:44 +00:00
public override void OnEntering ( ScreenTransitionEvent e )
2019-12-27 10:36:48 +00:00
{
2022-04-21 15:52:44 +00:00
base . OnEntering ( e ) ;
2019-12-27 10:36:48 +00:00
2021-01-04 09:32:23 +00:00
ApplyToBackground ( b = >
{
if ( epilepsyWarning ! = null )
epilepsyWarning . DimmableBackground = b ;
} ) ;
2020-10-25 11:33:35 +00:00
Beatmap . Value . Track . AddAdjustment ( AdjustableProperty . Volume , volumeAdjustment ) ;
2020-10-20 06:03:12 +00:00
2019-12-27 10:36:48 +00:00
content . ScaleTo ( 0.7f ) ;
contentIn ( ) ;
2020-01-08 18:55:35 +00:00
MetadataInfo . Delay ( 750 ) . FadeIn ( 500 ) ;
2020-12-10 07:33:28 +00:00
// after an initial delay, start the debounced load check.
// this will continue to execute even after resuming back on restart.
2021-11-12 11:30:40 +00:00
Scheduler . Add ( new ScheduledDelegate ( pushWhenLoaded , Clock . CurrentTime + PlayerPushDelay , 0 ) ) ;
2019-10-03 10:16:31 +00:00
2020-02-14 09:28:58 +00:00
showMuteWarningIfNeeded ( ) ;
2021-04-07 18:41:21 +00:00
showBatteryWarningIfNeeded ( ) ;
2019-10-03 10:16:31 +00:00
}
2018-08-02 10:47:50 +00:00
2022-04-21 15:52:44 +00:00
public override void OnResuming ( ScreenTransitionEvent e )
2017-04-18 04:30:51 +00:00
{
2022-04-21 15:52:44 +00:00
base . OnResuming ( e ) ;
2018-04-13 09:19:50 +00:00
2022-03-17 10:36:33 +00:00
Debug . Assert ( CurrentPlayer ! = null ) ;
2022-03-17 10:23:43 +00:00
var lastScore = CurrentPlayer . Score ;
2022-03-01 05:47:06 +00:00
2022-03-01 07:14:57 +00:00
AudioSettings . ReferenceScore . Value = lastScore ? . ScoreInfo ;
2022-03-01 05:47:06 +00:00
2020-12-10 07:56:56 +00:00
// prepare for a retry.
2022-03-17 10:23:43 +00:00
CurrentPlayer = null ;
2020-12-10 07:56:56 +00:00
playerConsumed = false ;
2020-12-10 07:34:58 +00:00
cancelLoad ( ) ;
2020-12-10 07:56:56 +00:00
2017-04-18 08:00:58 +00:00
contentIn ( ) ;
2017-04-18 04:30:51 +00:00
}
2018-04-13 09:19:50 +00:00
2022-04-21 15:52:44 +00:00
public override void OnSuspending ( ScreenTransitionEvent e )
2018-12-26 13:16:35 +00:00
{
2022-04-21 15:52:44 +00:00
base . OnSuspending ( e ) ;
2018-12-26 13:16:35 +00:00
2020-02-14 09:22:57 +00:00
BackgroundBrightnessReduction = false ;
2020-10-25 11:33:35 +00:00
// we're moving to player, so a period of silence is upcoming.
// stop the track before removing adjustment to avoid a volume spike.
Beatmap . Value . Track . Stop ( ) ;
Beatmap . Value . Track . RemoveAdjustment ( AdjustableProperty . Volume , volumeAdjustment ) ;
2021-10-07 11:52:36 +00:00
lowPassFilter . CutoffTo ( AudioFilter . MAX_LOWPASS_CUTOFF ) ;
2021-11-10 08:08:51 +00:00
highPassFilter . CutoffTo ( 0 ) ;
2018-12-26 13:16:35 +00:00
}
2022-04-21 15:52:44 +00:00
public override bool OnExiting ( ScreenExitEvent e )
2017-04-18 08:00:58 +00:00
{
2020-02-14 09:22:57 +00:00
cancelLoad ( ) ;
2021-11-12 11:30:40 +00:00
ContentOut ( ) ;
2018-04-13 09:19:50 +00:00
2021-10-18 04:30:37 +00:00
// If the load sequence was interrupted, the epilepsy warning may already be displayed (or in the process of being displayed).
epilepsyWarning ? . Hide ( ) ;
2021-10-13 04:26:20 +00:00
// Ensure the screen doesn't expire until all the outwards fade operations have completed.
2021-11-12 11:30:40 +00:00
this . Delay ( CONTENT_OUT_DURATION ) . FadeOut ( ) ;
2019-03-28 07:29:35 +00:00
2021-04-13 06:24:35 +00:00
ApplyToBackground ( b = > b . IgnoreUserSettings . Value = true ) ;
2021-01-04 09:32:23 +00:00
2020-02-14 09:22:57 +00:00
BackgroundBrightnessReduction = false ;
2020-10-25 11:33:35 +00:00
Beatmap . Value . Track . RemoveAdjustment ( AdjustableProperty . Volume , volumeAdjustment ) ;
2020-02-14 09:22:57 +00:00
2022-04-21 15:52:44 +00:00
return base . OnExiting ( e ) ;
2017-04-18 08:00:58 +00:00
}
2018-04-13 09:19:50 +00:00
2017-11-09 08:38:20 +00:00
protected override void LogoArriving ( OsuLogo logo , bool resuming )
2017-11-01 11:54:58 +00:00
{
2017-11-09 08:38:20 +00:00
base . LogoArriving ( logo , resuming ) ;
2018-04-13 09:19:50 +00:00
2019-03-22 11:01:58 +00:00
const double duration = 300 ;
2018-04-13 09:19:50 +00:00
2021-11-12 10:30:04 +00:00
if ( ! resuming ) logo . MoveTo ( new Vector2 ( 0.5f ) , duration , Easing . OutQuint ) ;
2019-03-28 07:09:42 +00:00
2021-11-12 10:30:04 +00:00
logo . ScaleTo ( new Vector2 ( 0.15f ) , duration , Easing . OutQuint ) ;
2017-11-02 12:52:01 +00:00
logo . FadeIn ( 350 ) ;
2018-04-13 09:19:50 +00:00
2019-04-24 07:20:51 +00:00
Scheduler . AddDelayed ( ( ) = >
{
if ( this . IsCurrentScreen ( ) )
content . StartTracking ( logo , resuming ? 0 : 500 , Easing . InOutExpo ) ;
} , resuming ? 0 : 500 ) ;
2019-03-26 08:18:35 +00:00
}
protected override void LogoExiting ( OsuLogo logo )
{
base . LogoExiting ( logo ) ;
2019-04-08 06:24:09 +00:00
content . StopTracking ( ) ;
2017-11-01 11:54:58 +00:00
}
2018-04-13 09:19:50 +00:00
2020-02-14 09:28:58 +00:00
#endregion
2020-02-14 09:22:57 +00:00
protected override void Update ( )
{
base . Update ( ) ;
2018-04-13 09:19:50 +00:00
2020-02-14 09:22:57 +00:00
if ( ! this . IsCurrentScreen ( ) )
return ;
// We need to perform this check here rather than in OnHover as any number of children of VisualSettings
// may also be handling the hover events.
if ( inputManager . HoveredDrawables . Contains ( VisualSettings ) )
{
// Preview user-defined background dim and blur when hovered on the visual settings panel.
2021-01-04 09:32:23 +00:00
ApplyToBackground ( b = >
{
2021-04-13 06:24:35 +00:00
b . IgnoreUserSettings . Value = false ;
2021-01-04 09:32:23 +00:00
b . BlurAmount . Value = 0 ;
} ) ;
2020-02-14 09:22:57 +00:00
BackgroundBrightnessReduction = false ;
}
else
{
2021-01-04 09:32:23 +00:00
ApplyToBackground ( b = >
{
// Returns background dim and blur to the values specified by PlayerLoader.
2021-04-13 06:24:35 +00:00
b . IgnoreUserSettings . Value = true ;
2021-01-04 09:32:23 +00:00
b . BlurAmount . Value = BACKGROUND_BLUR ;
} ) ;
2020-02-14 09:22:57 +00:00
BackgroundBrightnessReduction = true ;
}
}
2020-12-10 07:32:14 +00:00
private Player consumePlayer ( )
{
2020-12-10 07:56:56 +00:00
Debug . Assert ( ! playerConsumed ) ;
2022-03-17 10:36:33 +00:00
Debug . Assert ( CurrentPlayer ! = null ) ;
2020-12-10 07:56:56 +00:00
playerConsumed = true ;
2022-03-17 10:23:43 +00:00
return CurrentPlayer ;
2020-12-10 07:32:14 +00:00
}
2020-02-14 09:22:57 +00:00
private void prepareNewPlayer ( )
{
2020-08-13 03:04:32 +00:00
if ( ! this . IsCurrentScreen ( ) )
return ;
2022-03-17 10:23:43 +00:00
CurrentPlayer = createPlayer ( ) ;
2022-08-31 10:50:16 +00:00
CurrentPlayer . Configuration . AutomaticallySkipIntro | = quickRestart ;
2022-03-17 10:23:43 +00:00
CurrentPlayer . RestartCount = restartCount + + ;
CurrentPlayer . RestartRequested = restartRequested ;
2019-03-20 10:59:54 +00:00
2022-04-28 11:10:47 +00:00
LoadTask = LoadComponentAsync ( CurrentPlayer , _ = >
{
MetadataInfo . Loading = false ;
OnPlayerLoaded ( ) ;
} ) ;
}
protected virtual void OnPlayerLoaded ( )
{
2020-02-14 09:22:57 +00:00
}
2022-08-16 04:04:56 +00:00
private void restartRequested ( bool quickRestartRequested )
2020-02-14 09:22:57 +00:00
{
2022-08-16 04:04:56 +00:00
quickRestart = quickRestartRequested ;
2020-02-14 09:22:57 +00:00
hideOverlays = true ;
ValidForResume = true ;
}
private void contentIn ( )
{
2020-08-13 03:04:32 +00:00
MetadataInfo . Loading = true ;
2022-10-13 08:12:28 +00:00
settingsScroll . FadeInFromZero ( 400 ) ;
2020-02-14 09:22:57 +00:00
content . FadeInFromZero ( 400 ) ;
2022-08-06 13:12:36 +00:00
content . ScaleTo ( 1 , 650 , Easing . OutQuint ) . Then ( ) . Schedule ( prepareNewPlayer ) ;
2021-10-13 04:26:20 +00:00
lowPassFilter . CutoffTo ( 1000 , 650 , Easing . OutQuint ) ;
2021-11-12 05:22:43 +00:00
highPassFilter . CutoffTo ( 300 ) . Then ( ) . CutoffTo ( 0 , 1250 ) ; // 1250 is to line up with the appearance of MetadataInfo (750 delay + 500 fade-in)
2021-06-09 08:17:39 +00:00
ApplyToBackground ( b = > b ? . FadeColour ( Color4 . White , 800 , Easing . OutQuint ) ) ;
2020-02-14 09:22:57 +00:00
}
2021-11-12 11:30:40 +00:00
protected virtual void ContentOut ( )
2020-02-14 09:22:57 +00:00
{
// Ensure the logo is no longer tracking before we scale the content
content . StopTracking ( ) ;
2021-11-12 11:30:40 +00:00
content . ScaleTo ( 0.7f , CONTENT_OUT_DURATION * 2 , Easing . OutQuint ) ;
content . FadeOut ( CONTENT_OUT_DURATION , Easing . OutQuint ) ;
2022-10-13 08:12:28 +00:00
settingsScroll . FadeOut ( CONTENT_OUT_DURATION , Easing . OutQuint ) ;
2021-11-12 11:30:40 +00:00
lowPassFilter . CutoffTo ( AudioFilter . MAX_LOWPASS_CUTOFF , CONTENT_OUT_DURATION ) ;
highPassFilter . CutoffTo ( 0 , CONTENT_OUT_DURATION ) ;
2020-02-14 09:22:57 +00:00
}
2018-04-13 09:19:50 +00:00
2017-04-18 08:07:02 +00:00
private void pushWhenLoaded ( )
{
2019-01-23 11:52:00 +00:00
if ( ! this . IsCurrentScreen ( ) ) return ;
2018-04-13 09:19:50 +00:00
2020-12-10 07:33:28 +00:00
if ( ! readyForPush )
2017-09-14 23:15:23 +00:00
{
2020-12-10 07:33:28 +00:00
// as the pushDebounce below has a delay, we need to keep checking and cancel a future debounce
// if we become unready for push during the delay.
cancelLoad ( ) ;
return ;
}
2018-04-13 09:19:50 +00:00
2020-12-10 07:33:28 +00:00
// if a push has already been scheduled, no further action is required.
// this value is reset via cancelLoad() to allow a second usage of the same PlayerLoader screen.
if ( scheduledPushPlayer ! = null )
return ;
2018-04-13 09:19:50 +00:00
2020-12-10 07:33:28 +00:00
scheduledPushPlayer = Scheduler . AddDelayed ( ( ) = >
{
2020-12-10 07:32:14 +00:00
// ensure that once we have reached this "point of no return", readyForPush will be false for all future checks (until a new player instance is prepared).
var consumedPlayer = consumePlayer ( ) ;
2018-04-13 09:19:50 +00:00
2021-11-12 11:30:40 +00:00
ContentOut ( ) ;
2020-10-19 22:22:30 +00:00
2021-11-12 11:30:40 +00:00
TransformSequence < PlayerLoader > pushSequence = this . Delay ( CONTENT_OUT_DURATION ) ;
2020-10-19 21:32:44 +00:00
2020-12-10 07:33:28 +00:00
// only show if the warning was created (i.e. the beatmap needs it)
// and this is not a restart of the map (the warning expires after first load).
if ( epilepsyWarning ? . IsAlive = = true )
{
const double epilepsy_display_length = 3000 ;
pushSequence
. Schedule ( ( ) = > epilepsyWarning . State . Value = Visibility . Visible )
. TransformBindableTo ( volumeAdjustment , 0.25 , EpilepsyWarning . FADE_DURATION , Easing . OutQuint )
. Delay ( epilepsy_display_length )
. Schedule ( ( ) = >
{
epilepsyWarning . Hide ( ) ;
epilepsyWarning . Expire ( ) ;
} )
. Delay ( EpilepsyWarning . FADE_DURATION ) ;
}
2021-10-13 04:26:20 +00:00
else
{
// This goes hand-in-hand with the restoration of low pass filter in contentOut().
2021-11-12 11:30:40 +00:00
this . TransformBindableTo ( volumeAdjustment , 0 , CONTENT_OUT_DURATION , Easing . OutCubic ) ;
2021-10-13 04:26:20 +00:00
}
2018-04-13 09:19:50 +00:00
2020-12-10 07:33:28 +00:00
pushSequence . Schedule ( ( ) = >
{
if ( ! this . IsCurrentScreen ( ) ) return ;
2018-04-13 09:19:50 +00:00
2020-12-10 07:33:28 +00:00
LoadTask = null ;
// By default, we want to load the player and never be returned to.
// Note that this may change if the player we load requested a re-run.
ValidForResume = false ;
2018-04-20 07:52:15 +00:00
2020-12-10 07:32:14 +00:00
if ( consumedPlayer . LoadedBeatmapSuccessfully )
this . Push ( consumedPlayer ) ;
2020-12-10 07:33:28 +00:00
else
this . Exit ( ) ;
} ) ;
2022-08-06 13:12:36 +00:00
} , 500 ) ;
2017-02-22 12:43:29 +00:00
}
2018-04-13 09:19:50 +00:00
2018-03-22 04:01:40 +00:00
private void cancelLoad ( )
{
2020-02-14 09:22:57 +00:00
scheduledPushPlayer ? . Cancel ( ) ;
scheduledPushPlayer = null ;
2018-03-22 04:01:40 +00:00
}
2018-04-13 09:19:50 +00:00
2020-02-14 09:22:57 +00:00
#region Disposal
2018-04-13 09:19:50 +00:00
2018-03-23 06:04:40 +00:00
protected override void Dispose ( bool isDisposing )
{
base . Dispose ( isDisposing ) ;
2018-04-13 09:19:50 +00:00
2018-05-30 06:47:31 +00:00
if ( isDisposing )
{
// if the player never got pushed, we should explicitly dispose it.
2022-03-17 10:23:43 +00:00
DisposalTask = LoadTask ? . ContinueWith ( _ = > CurrentPlayer ? . Dispose ( ) ) ;
2018-05-30 06:47:31 +00:00
}
2018-03-23 06:04:40 +00:00
}
2018-04-13 09:19:50 +00:00
2020-02-14 09:22:57 +00:00
#endregion
2019-03-20 07:50:47 +00:00
2020-02-14 09:28:58 +00:00
#region Mute warning
2022-03-17 10:36:33 +00:00
private Bindable < bool > muteWarningShownOnce = null ! ;
2020-02-14 09:28:58 +00:00
2021-03-31 04:57:57 +00:00
private int restartCount ;
2022-09-20 03:46:35 +00:00
private const double volume_requirement = 0.01 ;
2021-12-08 04:38:13 +00:00
2020-02-14 09:28:58 +00:00
private void showMuteWarningIfNeeded ( )
{
if ( ! muteWarningShownOnce . Value )
{
2020-05-05 01:31:11 +00:00
// Checks if the notification has not been shown yet and also if master volume is muted, track/music volume is muted or if the whole game is muted.
2021-12-08 04:38:13 +00:00
if ( volumeOverlay ? . IsMuted . Value = = true | | audioManager . Volume . Value < = volume_requirement | | audioManager . VolumeTrack . Value < = volume_requirement )
2020-02-14 09:28:58 +00:00
{
notificationOverlay ? . Post ( new MutedNotification ( ) ) ;
muteWarningShownOnce . Value = true ;
}
}
}
2019-09-15 13:59:46 +00:00
private class MutedNotification : SimpleNotification
{
2020-02-14 09:22:57 +00:00
public override bool IsImportant = > true ;
2019-09-15 15:47:44 +00:00
public MutedNotification ( )
{
2021-12-08 04:38:13 +00:00
Text = "Your game volume is too low to hear anything! Click here to restore it." ;
2019-09-15 15:47:44 +00:00
}
2019-09-15 13:59:46 +00:00
[BackgroundDependencyLoader]
2022-04-18 10:59:57 +00:00
private void load ( OsuColour colours , AudioManager audioManager , INotificationOverlay notificationOverlay , VolumeOverlay volumeOverlay )
2019-09-15 13:59:46 +00:00
{
Icon = FontAwesome . Solid . VolumeMute ;
2022-08-30 08:40:35 +00:00
IconContent . Colour = colours . RedDark ;
2019-09-15 13:59:46 +00:00
Activated = delegate
{
notificationOverlay . Hide ( ) ;
2019-09-15 14:50:01 +00:00
volumeOverlay . IsMuted . Value = false ;
2021-12-08 04:38:13 +00:00
// Check values before resetting, as the user may have only had mute enabled, in which case we might not need to adjust volumes.
2022-09-20 03:52:04 +00:00
// Note that we only restore halfway to ensure the user isn't suddenly overloaded by unexpectedly high volume.
2021-12-09 09:58:47 +00:00
if ( audioManager . Volume . Value < = volume_requirement )
2022-09-20 03:52:04 +00:00
audioManager . Volume . Value = 0.5f ;
2021-12-09 09:58:47 +00:00
if ( audioManager . VolumeTrack . Value < = volume_requirement )
2022-09-20 03:52:04 +00:00
audioManager . VolumeTrack . Value = 0.5f ;
2019-09-15 13:59:46 +00:00
return true ;
} ;
}
}
2020-02-14 09:28:58 +00:00
#endregion
2021-04-07 18:41:21 +00:00
#region Low battery warning
2021-04-09 00:28:23 +00:00
2022-07-30 12:26:19 +00:00
private const double low_battery_threshold = 0.25 ;
2022-03-17 10:36:33 +00:00
private Bindable < bool > batteryWarningShownOnce = null ! ;
2021-04-07 18:41:21 +00:00
private void showBatteryWarningIfNeeded ( )
{
2021-04-12 14:52:12 +00:00
if ( batteryInfo = = null ) return ;
2021-04-09 21:55:41 +00:00
2021-04-07 18:41:21 +00:00
if ( ! batteryWarningShownOnce . Value )
{
2022-07-30 12:26:19 +00:00
if ( batteryInfo . OnBattery & & batteryInfo . ChargeLevel < = low_battery_threshold )
2021-04-07 18:41:21 +00:00
{
notificationOverlay ? . Post ( new BatteryWarningNotification ( ) ) ;
batteryWarningShownOnce . Value = true ;
}
}
}
private class BatteryWarningNotification : SimpleNotification
{
public override bool IsImportant = > true ;
public BatteryWarningNotification ( )
{
2021-04-09 21:55:41 +00:00
Text = "Your battery level is low! Charge your device to prevent interruptions during gameplay." ;
2021-04-07 18:41:21 +00:00
}
[BackgroundDependencyLoader]
2022-04-18 10:59:57 +00:00
private void load ( OsuColour colours , INotificationOverlay notificationOverlay )
2021-04-07 18:41:21 +00:00
{
Icon = FontAwesome . Solid . BatteryQuarter ;
2022-08-30 08:40:35 +00:00
IconContent . Colour = colours . RedDark ;
2021-04-07 18:41:21 +00:00
Activated = delegate
{
notificationOverlay . Hide ( ) ;
return true ;
} ;
}
}
#endregion
2017-02-22 12:43:29 +00:00
}
}