2017-02-07 12:59:30 +08:00
|
|
|
|
// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
|
|
|
|
|
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
|
2016-09-29 19:13:58 +08:00
|
|
|
|
|
2017-03-10 10:59:08 +08:00
|
|
|
|
using OpenTK;
|
2016-11-09 07:13:20 +08:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Audio;
|
2017-03-10 10:59:08 +08:00
|
|
|
|
using osu.Framework.Configuration;
|
2016-11-14 16:23:33 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
2017-03-10 10:59:08 +08:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Input;
|
|
|
|
|
using osu.Framework.Logging;
|
|
|
|
|
using osu.Framework.Screens;
|
2016-11-14 16:23:33 +08:00
|
|
|
|
using osu.Framework.Timing;
|
2017-03-10 10:59:08 +08:00
|
|
|
|
using osu.Game.Configuration;
|
2017-04-18 15:05:58 +08:00
|
|
|
|
using osu.Game.Rulesets;
|
|
|
|
|
using osu.Game.Rulesets.UI;
|
2017-03-10 10:59:08 +08:00
|
|
|
|
using osu.Game.Screens.Backgrounds;
|
2016-12-18 03:59:41 +08:00
|
|
|
|
using System;
|
2017-01-27 20:57:22 +08:00
|
|
|
|
using System.Linq;
|
2017-11-10 17:32:15 +08:00
|
|
|
|
using System.Threading;
|
|
|
|
|
using System.Threading.Tasks;
|
2017-04-07 15:55:41 +08:00
|
|
|
|
using osu.Framework.Threading;
|
2017-04-21 16:33:20 +08:00
|
|
|
|
using osu.Game.Rulesets.Mods;
|
2017-04-18 15:05:58 +08:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2017-04-11 13:01:13 +08:00
|
|
|
|
using osu.Game.Screens.Ranking;
|
2017-06-29 01:19:04 +08:00
|
|
|
|
using osu.Framework.Audio.Sample;
|
2017-08-05 15:22:10 +08:00
|
|
|
|
using osu.Game.Beatmaps;
|
2017-08-22 15:58:47 +08:00
|
|
|
|
using osu.Game.Online.API;
|
2017-09-21 03:33:07 +08:00
|
|
|
|
using osu.Game.Screens.Play.BreaksOverlay;
|
2017-09-14 20:28:53 +08:00
|
|
|
|
using osu.Game.Storyboards.Drawables;
|
|
|
|
|
using OpenTK.Graphics;
|
2016-09-29 19:13:58 +08:00
|
|
|
|
|
2016-11-14 16:23:33 +08:00
|
|
|
|
namespace osu.Game.Screens.Play
|
2016-09-29 19:13:58 +08:00
|
|
|
|
{
|
2017-02-17 17:59:30 +08:00
|
|
|
|
public class Player : OsuScreen
|
2016-09-29 19:13:58 +08:00
|
|
|
|
{
|
2017-02-17 17:59:30 +08:00
|
|
|
|
protected override BackgroundScreen CreateBackground() => new BackgroundScreenBeatmap(Beatmap);
|
2016-10-05 19:03:52 +08:00
|
|
|
|
|
2017-11-01 15:57:59 +08:00
|
|
|
|
public override bool ShowOverlays => false;
|
2016-11-09 14:22:54 +08:00
|
|
|
|
|
2017-11-01 15:57:59 +08:00
|
|
|
|
public override bool HasLocalCursorDisplayed => !pauseContainer.IsPaused && !HasFailed && RulesetContainer.ProvidingUserCursor;
|
2017-03-16 22:58:36 +08:00
|
|
|
|
|
2017-04-18 15:56:01 +08:00
|
|
|
|
public Action RestartRequested;
|
2017-04-18 12:30:51 +08:00
|
|
|
|
|
2017-11-01 15:57:59 +08:00
|
|
|
|
public override bool AllowBeatmapRulesetChange => false;
|
2017-04-21 15:03:59 +08:00
|
|
|
|
|
2017-04-02 02:17:24 +08:00
|
|
|
|
public bool HasFailed { get; private set; }
|
2017-03-28 09:53:57 +08:00
|
|
|
|
|
2017-01-30 16:08:14 +08:00
|
|
|
|
public int RestartCount;
|
2017-01-28 03:28:39 +08:00
|
|
|
|
|
2017-04-26 17:16:55 +08:00
|
|
|
|
private IAdjustableClock adjustableSourceClock;
|
|
|
|
|
private FramedOffsetClock offsetClock;
|
|
|
|
|
private DecoupleableInterpolatingFramedClock decoupledClock;
|
2016-11-16 14:48:35 +08:00
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
private PauseContainer pauseContainer;
|
|
|
|
|
|
2017-04-20 10:16:08 +08:00
|
|
|
|
private RulesetInfo ruleset;
|
2016-10-28 13:14:45 +08:00
|
|
|
|
|
2017-08-22 15:58:47 +08:00
|
|
|
|
private APIAccess api;
|
|
|
|
|
|
2016-11-29 19:30:16 +08:00
|
|
|
|
private ScoreProcessor scoreProcessor;
|
2017-08-09 12:28:29 +08:00
|
|
|
|
protected RulesetContainer RulesetContainer;
|
2017-04-21 15:19:40 +08:00
|
|
|
|
|
|
|
|
|
#region User Settings
|
|
|
|
|
|
2017-04-20 20:43:30 +08:00
|
|
|
|
private Bindable<double> dimLevel;
|
2017-09-14 21:44:36 +08:00
|
|
|
|
private Bindable<bool> showStoryboard;
|
2017-04-21 15:19:40 +08:00
|
|
|
|
private Bindable<bool> mouseWheelDisabled;
|
|
|
|
|
private Bindable<double> userAudioOffset;
|
|
|
|
|
|
2017-06-29 01:19:04 +08:00
|
|
|
|
private SampleChannel sampleRestart;
|
|
|
|
|
|
2017-04-21 15:19:40 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
2017-09-23 21:42:18 +08:00
|
|
|
|
private BreakOverlay breakOverlay;
|
2017-09-14 22:50:10 +08:00
|
|
|
|
private Container storyboardContainer;
|
2017-09-14 20:28:53 +08:00
|
|
|
|
private DrawableStoryboard storyboard;
|
|
|
|
|
|
2017-05-05 12:00:05 +08:00
|
|
|
|
private HUDOverlay hudOverlay;
|
2017-03-28 15:49:58 +08:00
|
|
|
|
private FailOverlay failOverlay;
|
2017-01-28 03:28:39 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
private bool loadedSuccessfully => RulesetContainer?.Objects.Any() == true;
|
2017-07-19 18:10:04 +08:00
|
|
|
|
|
2017-08-22 17:02:38 +08:00
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(AudioManager audio, OsuConfigManager config, APIAccess api)
|
2016-10-05 19:49:31 +08:00
|
|
|
|
{
|
2017-08-22 15:58:47 +08:00
|
|
|
|
this.api = api;
|
|
|
|
|
|
2017-05-15 09:56:27 +08:00
|
|
|
|
dimLevel = config.GetBindable<double>(OsuSetting.DimLevel);
|
2017-09-14 21:44:36 +08:00
|
|
|
|
showStoryboard = config.GetBindable<bool>(OsuSetting.ShowStoryboard);
|
2017-08-22 15:58:47 +08:00
|
|
|
|
|
2017-05-15 09:56:27 +08:00
|
|
|
|
mouseWheelDisabled = config.GetBindable<bool>(OsuSetting.MouseDisableWheel);
|
2017-02-28 18:44:12 +08:00
|
|
|
|
|
2017-06-29 01:19:04 +08:00
|
|
|
|
sampleRestart = audio.Sample.Get(@"Gameplay/restart");
|
|
|
|
|
|
2017-08-05 15:22:10 +08:00
|
|
|
|
WorkingBeatmap working = Beatmap.Value;
|
|
|
|
|
Beatmap beatmap;
|
|
|
|
|
|
2016-10-27 19:37:01 +08:00
|
|
|
|
try
|
|
|
|
|
{
|
2017-08-05 15:22:10 +08:00
|
|
|
|
beatmap = working.Beatmap;
|
|
|
|
|
|
|
|
|
|
if (beatmap == null)
|
2017-05-07 00:38:17 +08:00
|
|
|
|
throw new InvalidOperationException("Beatmap was not loaded");
|
2017-04-17 14:44:46 +08:00
|
|
|
|
|
2017-08-22 17:02:38 +08:00
|
|
|
|
ruleset = Ruleset.Value ?? beatmap.BeatmapInfo.Ruleset;
|
2017-08-21 11:31:21 +08:00
|
|
|
|
var rulesetInstance = ruleset.CreateInstance();
|
2017-04-20 10:36:50 +08:00
|
|
|
|
|
2017-04-17 14:44:46 +08:00
|
|
|
|
try
|
|
|
|
|
{
|
2017-08-09 12:28:29 +08:00
|
|
|
|
RulesetContainer = rulesetInstance.CreateRulesetContainerWith(working, ruleset.ID == beatmap.BeatmapInfo.Ruleset.ID);
|
2017-04-17 14:44:46 +08:00
|
|
|
|
}
|
2017-04-20 11:11:03 +08:00
|
|
|
|
catch (BeatmapInvalidForRulesetException)
|
2017-04-17 14:44:46 +08:00
|
|
|
|
{
|
2017-08-09 12:28:29 +08:00
|
|
|
|
// we may fail to create a RulesetContainer if the beatmap cannot be loaded with the user's preferred ruleset
|
2017-04-20 10:36:50 +08:00
|
|
|
|
// let's try again forcing the beatmap's ruleset.
|
2017-08-05 15:22:10 +08:00
|
|
|
|
ruleset = beatmap.BeatmapInfo.Ruleset;
|
2017-04-20 10:36:50 +08:00
|
|
|
|
rulesetInstance = ruleset.CreateInstance();
|
2017-08-09 12:28:29 +08:00
|
|
|
|
RulesetContainer = rulesetInstance.CreateRulesetContainerWith(Beatmap, true);
|
2017-04-17 14:44:46 +08:00
|
|
|
|
}
|
2017-04-26 19:22:03 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
if (!RulesetContainer.Objects.Any())
|
2017-05-07 00:38:17 +08:00
|
|
|
|
throw new InvalidOperationException("Beatmap contains no hit objects!");
|
2016-10-27 19:37:01 +08:00
|
|
|
|
}
|
2017-02-15 11:38:10 +08:00
|
|
|
|
catch (Exception e)
|
2016-10-27 19:37:01 +08:00
|
|
|
|
{
|
2017-02-15 11:38:10 +08:00
|
|
|
|
Logger.Log($"Could not load this beatmap sucessfully ({e})!", LoggingTarget.Runtime, LogLevel.Error);
|
|
|
|
|
|
2016-10-27 19:37:01 +08:00
|
|
|
|
//couldn't load, hard abort!
|
|
|
|
|
Exit();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2016-10-05 19:49:31 +08:00
|
|
|
|
|
2017-08-05 15:22:10 +08:00
|
|
|
|
adjustableSourceClock = (IAdjustableClock)working.Track ?? new StopwatchClock();
|
2017-04-26 17:32:47 +08:00
|
|
|
|
decoupledClock = new DecoupleableInterpolatingFramedClock { IsCoupled = false };
|
2017-04-26 17:16:55 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
var firstObjectTime = RulesetContainer.Objects.First().StartTime;
|
2017-08-05 15:22:10 +08:00
|
|
|
|
decoupledClock.Seek(Math.Min(0, firstObjectTime - Math.Max(beatmap.ControlPointInfo.TimingPointAt(firstObjectTime).BeatLength * 4, beatmap.BeatmapInfo.AudioLeadIn)));
|
2017-04-26 17:07:22 +08:00
|
|
|
|
decoupledClock.ProcessFrame();
|
|
|
|
|
|
2017-04-26 17:16:55 +08:00
|
|
|
|
offsetClock = new FramedOffsetClock(decoupledClock);
|
2017-04-21 15:19:40 +08:00
|
|
|
|
|
2017-05-15 09:56:27 +08:00
|
|
|
|
userAudioOffset = config.GetBindable<double>(OsuSetting.AudioOffset);
|
2017-04-21 15:19:40 +08:00
|
|
|
|
userAudioOffset.ValueChanged += v => offsetClock.Offset = v;
|
|
|
|
|
userAudioOffset.TriggerChange();
|
|
|
|
|
|
2016-10-19 18:44:03 +08:00
|
|
|
|
Children = new Drawable[]
|
2016-10-06 22:33:09 +08:00
|
|
|
|
{
|
2017-09-14 22:50:10 +08:00
|
|
|
|
storyboardContainer = new Container
|
2017-09-14 20:28:53 +08:00
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
Clock = offsetClock,
|
2017-09-15 17:23:37 +08:00
|
|
|
|
Alpha = 0,
|
2017-09-14 20:28:53 +08:00
|
|
|
|
},
|
2017-05-17 20:57:01 +08:00
|
|
|
|
pauseContainer = new PauseContainer
|
2016-11-09 17:50:30 +08:00
|
|
|
|
{
|
2017-05-17 20:57:01 +08:00
|
|
|
|
AudioClock = decoupledClock,
|
|
|
|
|
FramedClock = offsetClock,
|
|
|
|
|
OnRetry = Restart,
|
|
|
|
|
OnQuit = Exit,
|
2017-08-09 12:28:29 +08:00
|
|
|
|
CheckCanPause = () => ValidForResume && !HasFailed && !RulesetContainer.HasReplayLoaded,
|
2017-05-17 20:57:01 +08:00
|
|
|
|
Retries = RestartCount,
|
|
|
|
|
OnPause = () => {
|
|
|
|
|
hudOverlay.KeyCounter.IsCounting = pauseContainer.IsPaused;
|
|
|
|
|
},
|
|
|
|
|
OnResume = () => {
|
|
|
|
|
hudOverlay.KeyCounter.IsCounting = true;
|
|
|
|
|
},
|
2016-11-09 17:50:30 +08:00
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
2017-05-19 21:12:09 +08:00
|
|
|
|
new SkipButton(firstObjectTime) { AudioClock = decoupledClock },
|
2017-04-26 17:07:22 +08:00
|
|
|
|
new Container
|
2017-02-28 19:14:48 +08:00
|
|
|
|
{
|
2017-04-26 17:07:22 +08:00
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
Clock = offsetClock,
|
2017-09-16 00:39:55 +08:00
|
|
|
|
Child = RulesetContainer,
|
2017-02-28 19:14:48 +08:00
|
|
|
|
},
|
2017-06-05 16:15:04 +08:00
|
|
|
|
hudOverlay = new HUDOverlay
|
2017-05-17 20:57:01 +08:00
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre
|
|
|
|
|
},
|
2017-09-23 21:42:18 +08:00
|
|
|
|
breakOverlay = new BreakOverlay(beatmap.BeatmapInfo.LetterboxInBreaks)
|
2017-09-16 00:39:55 +08:00
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
2017-10-02 14:04:03 +08:00
|
|
|
|
Clock = decoupledClock,
|
|
|
|
|
Breaks = beatmap.Breaks
|
2017-09-16 00:39:55 +08:00
|
|
|
|
},
|
2016-11-09 17:50:30 +08:00
|
|
|
|
}
|
|
|
|
|
},
|
2017-04-06 14:34:52 +08:00
|
|
|
|
failOverlay = new FailOverlay
|
|
|
|
|
{
|
|
|
|
|
OnRetry = Restart,
|
|
|
|
|
OnQuit = Exit,
|
2017-04-09 21:26:31 +08:00
|
|
|
|
},
|
2017-04-10 11:06:10 +08:00
|
|
|
|
new HotkeyRetryOverlay
|
2017-04-09 21:26:31 +08:00
|
|
|
|
{
|
2017-04-11 18:58:57 +08:00
|
|
|
|
Action = () => {
|
|
|
|
|
//we want to hide the hitrenderer immediately (looks better).
|
|
|
|
|
//we may be able to remove this once the mouse cursor trail is improved.
|
2017-08-09 12:28:29 +08:00
|
|
|
|
RulesetContainer?.Hide();
|
2017-04-11 18:58:57 +08:00
|
|
|
|
Restart();
|
|
|
|
|
},
|
2017-04-06 14:34:52 +08:00
|
|
|
|
}
|
2016-10-19 18:44:03 +08:00
|
|
|
|
};
|
2016-10-28 13:14:45 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
scoreProcessor = RulesetContainer.CreateScoreProcessor();
|
2017-05-17 20:57:01 +08:00
|
|
|
|
|
2017-09-14 22:50:10 +08:00
|
|
|
|
if (showStoryboard)
|
2017-09-15 00:56:01 +08:00
|
|
|
|
initializeStoryboard(false);
|
2017-09-14 20:28:53 +08:00
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
hudOverlay.BindProcessor(scoreProcessor);
|
2017-08-09 12:28:29 +08:00
|
|
|
|
hudOverlay.BindRulesetContainer(RulesetContainer);
|
2017-05-16 16:55:35 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
hudOverlay.Progress.Objects = RulesetContainer.Objects;
|
2017-05-17 20:57:01 +08:00
|
|
|
|
hudOverlay.Progress.AudioClock = decoupledClock;
|
2017-08-09 12:28:29 +08:00
|
|
|
|
hudOverlay.Progress.AllowSeeking = RulesetContainer.HasReplayLoaded;
|
2017-05-17 20:57:01 +08:00
|
|
|
|
hudOverlay.Progress.OnSeek = pos => decoupledClock.Seek(pos);
|
|
|
|
|
|
2017-08-05 15:22:10 +08:00
|
|
|
|
hudOverlay.ModDisplay.Current.BindTo(working.Mods);
|
2017-05-17 20:57:01 +08:00
|
|
|
|
|
2017-09-23 21:42:18 +08:00
|
|
|
|
breakOverlay.BindProcessor(scoreProcessor);
|
|
|
|
|
|
2017-10-04 01:05:50 +08:00
|
|
|
|
hudOverlay.ReplaySettingsOverlay.PlaybackSettings.AdjustableClock = adjustableSourceClock;
|
2017-10-02 09:42:38 +08:00
|
|
|
|
|
2017-09-12 21:01:08 +08:00
|
|
|
|
// Bind ScoreProcessor to ourselves
|
|
|
|
|
scoreProcessor.AllJudged += onCompletion;
|
2017-05-17 20:57:01 +08:00
|
|
|
|
scoreProcessor.Failed += onFail;
|
2017-05-16 16:55:35 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-11-11 12:00:54 +08:00
|
|
|
|
private void applyRateFromMods()
|
|
|
|
|
{
|
2017-11-15 19:03:18 +08:00
|
|
|
|
if (adjustableSourceClock == null) return;
|
|
|
|
|
|
2017-11-11 12:00:54 +08:00
|
|
|
|
adjustableSourceClock.Rate = 1;
|
|
|
|
|
foreach (var mod in Beatmap.Value.Mods.Value.OfType<IApplicableToClock>())
|
|
|
|
|
mod.ApplyToClock(adjustableSourceClock);
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-15 00:56:01 +08:00
|
|
|
|
private void initializeStoryboard(bool asyncLoad)
|
2017-09-14 22:50:10 +08:00
|
|
|
|
{
|
|
|
|
|
var beatmap = Beatmap.Value.Beatmap;
|
|
|
|
|
|
2017-09-25 17:03:57 +08:00
|
|
|
|
storyboard = beatmap.Storyboard.CreateDrawable(Beatmap.Value);
|
2017-09-14 22:50:10 +08:00
|
|
|
|
storyboard.Masking = true;
|
2017-09-15 00:56:01 +08:00
|
|
|
|
|
|
|
|
|
storyboardContainer.Add(asyncLoad ? new AsyncLoadWrapper(storyboard) { RelativeSizeAxes = Axes.Both } : (Drawable)storyboard);
|
2017-09-14 22:50:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-01-29 04:55:42 +08:00
|
|
|
|
public void Restart()
|
|
|
|
|
{
|
2017-06-29 01:19:04 +08:00
|
|
|
|
sampleRestart?.Play();
|
2017-04-18 14:52:38 +08:00
|
|
|
|
ValidForResume = false;
|
2017-04-18 15:56:01 +08:00
|
|
|
|
RestartRequested?.Invoke();
|
2017-04-18 12:30:51 +08:00
|
|
|
|
Exit();
|
2017-01-28 03:28:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-07 15:55:41 +08:00
|
|
|
|
private ScheduledDelegate onCompletionEvent;
|
|
|
|
|
|
2017-03-16 12:39:57 +08:00
|
|
|
|
private void onCompletion()
|
2016-11-29 22:59:56 +08:00
|
|
|
|
{
|
2017-03-17 00:36:30 +08:00
|
|
|
|
// Only show the completion screen if the player hasn't failed
|
2017-04-07 15:55:41 +08:00
|
|
|
|
if (scoreProcessor.HasFailed || onCompletionEvent != null)
|
2017-03-16 23:30:23 +08:00
|
|
|
|
return;
|
|
|
|
|
|
2017-04-13 10:41:08 +08:00
|
|
|
|
ValidForResume = false;
|
|
|
|
|
|
2017-05-19 12:53:09 +08:00
|
|
|
|
using (BeginDelayedSequence(1000))
|
2016-11-29 22:59:56 +08:00
|
|
|
|
{
|
2017-05-19 12:53:09 +08:00
|
|
|
|
onCompletionEvent = Schedule(delegate
|
2017-04-20 10:16:08 +08:00
|
|
|
|
{
|
2017-05-19 12:53:09 +08:00
|
|
|
|
var score = new Score
|
|
|
|
|
{
|
2017-07-19 12:32:16 +08:00
|
|
|
|
Beatmap = Beatmap.Value.BeatmapInfo,
|
2017-05-19 12:53:09 +08:00
|
|
|
|
Ruleset = ruleset
|
|
|
|
|
};
|
|
|
|
|
scoreProcessor.PopulateScore(score);
|
2017-08-22 15:58:47 +08:00
|
|
|
|
score.User = RulesetContainer.Replay?.User ?? api.LocalUser.Value;
|
2017-05-19 12:53:09 +08:00
|
|
|
|
Push(new Results(score));
|
|
|
|
|
});
|
|
|
|
|
}
|
2016-11-29 22:59:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-05 10:59:58 +08:00
|
|
|
|
private bool onFail()
|
2017-01-20 15:51:43 +08:00
|
|
|
|
{
|
2017-08-05 10:59:58 +08:00
|
|
|
|
if (Beatmap.Value.Mods.Value.Any(m => !m.AllowFail))
|
|
|
|
|
return false;
|
|
|
|
|
|
2017-04-26 17:16:55 +08:00
|
|
|
|
decoupledClock.Stop();
|
2017-01-20 15:51:43 +08:00
|
|
|
|
|
2017-04-02 02:17:24 +08:00
|
|
|
|
HasFailed = true;
|
2017-03-28 09:53:57 +08:00
|
|
|
|
failOverlay.Retries = RestartCount;
|
|
|
|
|
failOverlay.Show();
|
2017-08-05 10:59:58 +08:00
|
|
|
|
return true;
|
2017-01-20 15:51:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-17 17:59:30 +08:00
|
|
|
|
protected override void OnEntering(Screen last)
|
2016-11-20 00:39:43 +08:00
|
|
|
|
{
|
|
|
|
|
base.OnEntering(last);
|
2017-02-15 11:38:10 +08:00
|
|
|
|
|
2017-07-19 18:10:04 +08:00
|
|
|
|
if (!loadedSuccessfully)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-07-23 02:50:25 +08:00
|
|
|
|
(Background as BackgroundScreenBeatmap)?.BlurTo(Vector2.Zero, 1500, Easing.OutQuint);
|
2016-11-20 00:39:43 +08:00
|
|
|
|
|
2017-09-14 21:58:55 +08:00
|
|
|
|
dimLevel.ValueChanged += dimLevel_ValueChanged;
|
|
|
|
|
showStoryboard.ValueChanged += showStoryboard_ValueChanged;
|
2017-09-14 21:44:36 +08:00
|
|
|
|
updateBackgroundElements();
|
2017-02-22 13:14:37 +08:00
|
|
|
|
|
2017-09-14 20:28:53 +08:00
|
|
|
|
Content.Alpha = 0;
|
2017-07-16 23:28:20 +08:00
|
|
|
|
Content
|
|
|
|
|
.ScaleTo(0.7f)
|
2017-07-23 02:50:25 +08:00
|
|
|
|
.ScaleTo(1, 750, Easing.OutQuint)
|
2017-07-16 23:28:20 +08:00
|
|
|
|
.Delay(250)
|
|
|
|
|
.FadeIn(250);
|
2017-02-22 13:14:37 +08:00
|
|
|
|
|
2017-11-13 18:43:00 +08:00
|
|
|
|
Task.Run(() =>
|
2017-07-16 22:37:59 +08:00
|
|
|
|
{
|
2017-11-13 18:43:00 +08:00
|
|
|
|
adjustableSourceClock.Reset();
|
|
|
|
|
|
|
|
|
|
// this is temporary until we have blocking (async.Wait()) audio component methods.
|
|
|
|
|
// then we can call ResetAsync().Wait() or the blocking version above.
|
|
|
|
|
while (adjustableSourceClock.IsRunning)
|
|
|
|
|
Thread.Sleep(1);
|
|
|
|
|
|
|
|
|
|
Schedule(() =>
|
|
|
|
|
{
|
|
|
|
|
decoupledClock.ChangeSource(adjustableSourceClock);
|
|
|
|
|
applyRateFromMods();
|
|
|
|
|
|
|
|
|
|
this.Delay(750).Schedule(() =>
|
|
|
|
|
{
|
|
|
|
|
if (!pauseContainer.IsPaused)
|
|
|
|
|
decoupledClock.Start();
|
|
|
|
|
});
|
|
|
|
|
});
|
2017-07-16 22:37:59 +08:00
|
|
|
|
});
|
2017-04-07 19:38:57 +08:00
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
pauseContainer.Alpha = 0;
|
2017-07-23 02:50:25 +08:00
|
|
|
|
pauseContainer.FadeIn(750, Easing.OutQuint);
|
2017-02-22 13:14:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void OnSuspending(Screen next)
|
|
|
|
|
{
|
2017-04-11 18:58:57 +08:00
|
|
|
|
fadeOut();
|
2017-02-22 13:14:37 +08:00
|
|
|
|
base.OnSuspending(next);
|
2016-10-28 13:14:45 +08:00
|
|
|
|
}
|
2016-11-16 14:48:35 +08:00
|
|
|
|
|
2017-02-17 17:59:30 +08:00
|
|
|
|
protected override bool OnExiting(Screen next)
|
2016-12-17 00:13:24 +08:00
|
|
|
|
{
|
2017-08-09 12:28:29 +08:00
|
|
|
|
if (HasFailed || !ValidForResume || pauseContainer?.AllowExit != false || RulesetContainer?.HasReplayLoaded != false)
|
2017-04-07 19:46:35 +08:00
|
|
|
|
{
|
2017-11-11 12:00:54 +08:00
|
|
|
|
// In the case of replays, we may have changed the playback rate.
|
|
|
|
|
applyRateFromMods();
|
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
fadeOut();
|
|
|
|
|
return base.OnExiting(next);
|
2017-04-07 19:46:35 +08:00
|
|
|
|
}
|
2017-03-04 20:35:26 +08:00
|
|
|
|
|
2017-07-19 18:10:04 +08:00
|
|
|
|
if (loadedSuccessfully)
|
|
|
|
|
{
|
|
|
|
|
pauseContainer.Pause();
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
return true;
|
2016-12-17 00:13:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-14 21:58:55 +08:00
|
|
|
|
private void dimLevel_ValueChanged(double newValue)
|
|
|
|
|
=> updateBackgroundElements();
|
|
|
|
|
|
|
|
|
|
private void showStoryboard_ValueChanged(bool newValue)
|
|
|
|
|
=> updateBackgroundElements();
|
|
|
|
|
|
2017-09-14 21:44:36 +08:00
|
|
|
|
private void updateBackgroundElements()
|
2017-09-14 20:28:53 +08:00
|
|
|
|
{
|
|
|
|
|
var opacity = 1 - (float)dimLevel;
|
|
|
|
|
|
2017-09-14 22:50:10 +08:00
|
|
|
|
if (showStoryboard && storyboard == null)
|
2017-09-15 00:56:01 +08:00
|
|
|
|
initializeStoryboard(true);
|
2017-09-14 22:50:10 +08:00
|
|
|
|
|
2017-09-15 17:23:37 +08:00
|
|
|
|
var beatmap = Beatmap.Value;
|
|
|
|
|
var storyboardVisible = showStoryboard && beatmap.Beatmap.Storyboard.HasDrawable;
|
2017-09-14 22:50:10 +08:00
|
|
|
|
|
2017-09-15 17:23:37 +08:00
|
|
|
|
storyboardContainer.FadeColour(new Color4(opacity, opacity, opacity, 1), 800);
|
|
|
|
|
storyboardContainer.FadeTo(storyboardVisible && opacity > 0 ? 1 : 0);
|
|
|
|
|
|
|
|
|
|
Background?.FadeTo(!storyboardVisible || beatmap.Background == null ? opacity : 0, 800, Easing.OutQuint);
|
2017-09-14 20:28:53 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-11 18:58:57 +08:00
|
|
|
|
private void fadeOut()
|
|
|
|
|
{
|
2017-09-14 21:58:55 +08:00
|
|
|
|
dimLevel.ValueChanged -= dimLevel_ValueChanged;
|
|
|
|
|
showStoryboard.ValueChanged -= showStoryboard_ValueChanged;
|
|
|
|
|
|
2017-04-11 18:58:57 +08:00
|
|
|
|
const float fade_out_duration = 250;
|
2017-02-15 11:38:10 +08:00
|
|
|
|
|
2017-08-09 12:28:29 +08:00
|
|
|
|
RulesetContainer?.FadeOut(fade_out_duration);
|
2017-04-11 18:58:57 +08:00
|
|
|
|
Content.FadeOut(fade_out_duration);
|
2017-03-04 20:35:26 +08:00
|
|
|
|
|
2017-07-23 02:50:25 +08:00
|
|
|
|
hudOverlay?.ScaleTo(0.7f, fade_out_duration * 3, Easing.In);
|
2017-02-01 08:44:59 +08:00
|
|
|
|
|
2017-04-11 18:58:57 +08:00
|
|
|
|
Background?.FadeTo(1f, fade_out_duration);
|
2016-12-17 00:13:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-17 20:57:01 +08:00
|
|
|
|
protected override bool OnWheel(InputState state) => mouseWheelDisabled.Value && !pauseContainer.IsPaused;
|
2016-09-29 19:13:58 +08:00
|
|
|
|
}
|
2017-04-11 23:09:45 +08:00
|
|
|
|
}
|