osu/osu.Game/Screens/Play/Player.cs

396 lines
14 KiB
C#
Raw Normal View History

2018-01-05 11:21:19 +00:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
2016-09-29 11:13:58 +00:00
2018-01-21 15:11:43 +00:00
using System;
using System.Linq;
using System.Threading.Tasks;
2016-11-08 23:13:20 +00:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2018-01-21 15:11:43 +00:00
using osu.Framework.Audio.Sample;
using osu.Framework.Configuration;
2016-11-14 08:23:33 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2018-01-21 15:11:43 +00:00
using osu.Framework.Graphics.Cursor;
using osu.Framework.Input;
using osu.Framework.Logging;
using osu.Framework.Screens;
using osu.Framework.Threading;
2016-11-14 08:23:33 +00:00
using osu.Framework.Timing;
using osu.Game.Beatmaps;
using osu.Game.Configuration;
2018-01-09 13:21:15 +00:00
using osu.Game.Graphics;
using osu.Game.Graphics.Cursor;
using osu.Game.Online.API;
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets;
using osu.Game.Rulesets.Mods;
2017-04-18 07:05:58 +00:00
using osu.Game.Rulesets.Scoring;
2018-01-21 15:11:43 +00:00
using osu.Game.Rulesets.UI;
using osu.Game.Screens.Ranking;
2018-03-20 07:26:36 +00:00
using osu.Game.Skinning;
2017-09-14 12:28:53 +00:00
using osu.Game.Storyboards.Drawables;
2016-09-29 11:13:58 +00:00
2016-11-14 08:23:33 +00:00
namespace osu.Game.Screens.Play
2016-09-29 11:13:58 +00:00
{
public class Player : ScreenWithBeatmapBackground, IProvideCursor
2016-09-29 11:13:58 +00:00
{
protected override float BackgroundParallaxAmount => 0.1f;
public override bool ShowOverlaysOnEnter => false;
public Action RestartRequested;
2017-04-01 18:17:24 +00:00
public bool HasFailed { get; private set; }
public bool AllowPause { get; set; } = true;
public bool AllowLeadIn { get; set; } = true;
public bool AllowResults { get; set; } = true;
private Bindable<bool> mouseWheelDisabled;
private Bindable<double> userAudioOffset;
public int RestartCount;
2018-01-12 10:36:35 +00:00
public CursorContainer Cursor => RulesetContainer.Cursor;
public bool ProvidingUserCursor => RulesetContainer?.Cursor != null && !RulesetContainer.HasReplayLoaded.Value;
2018-02-27 07:05:21 +00:00
private IAdjustableClock sourceClock;
/// <summary>
/// The decoupled clock used for gameplay. Should be used for seeks and clock control.
/// </summary>
private DecoupleableInterpolatingFramedClock adjustableClock;
2016-11-16 06:48:35 +00:00
private PauseContainer pauseContainer;
private RulesetInfo ruleset;
private APIAccess api;
private SampleChannel sampleRestart;
2016-11-29 11:30:16 +00:00
private ScoreProcessor scoreProcessor;
2017-08-09 04:28:29 +00:00
protected RulesetContainer RulesetContainer;
2017-09-14 12:28:53 +00:00
private HUDOverlay hudOverlay;
2017-03-28 07:49:58 +00:00
private FailOverlay failOverlay;
private DrawableStoryboard storyboard;
private Container storyboardContainer;
2017-08-09 04:28:29 +00:00
private bool loadedSuccessfully => RulesetContainer?.Objects.Any() == true;
2017-07-19 10:10:04 +00:00
2017-08-22 09:02:38 +00:00
[BackgroundDependencyLoader]
private void load(AudioManager audio, APIAccess api, OsuConfigManager config)
2016-10-05 11:49:31 +00:00
{
this.api = api;
sampleRestart = audio.Sample.Get(@"Gameplay/restart");
2017-05-15 01:56:27 +00:00
mouseWheelDisabled = config.GetBindable<bool>(OsuSetting.MouseDisableWheel);
userAudioOffset = config.GetBindable<double>(OsuSetting.AudioOffset);
WorkingBeatmap working = Beatmap.Value;
Beatmap beatmap;
try
{
beatmap = working.Beatmap;
if (beatmap == null)
2017-05-06 16:38:17 +00:00
throw new InvalidOperationException("Beatmap was not loaded");
2017-04-17 06:44:46 +00:00
2017-08-22 09:02:38 +00:00
ruleset = Ruleset.Value ?? beatmap.BeatmapInfo.Ruleset;
var rulesetInstance = ruleset.CreateInstance();
2017-04-20 02:36:50 +00:00
2017-04-17 06:44:46 +00:00
try
{
2017-08-09 04:28:29 +00:00
RulesetContainer = rulesetInstance.CreateRulesetContainerWith(working, ruleset.ID == beatmap.BeatmapInfo.Ruleset.ID);
2017-04-17 06:44:46 +00:00
}
2017-04-20 03:11:03 +00:00
catch (BeatmapInvalidForRulesetException)
2017-04-17 06:44:46 +00:00
{
2017-08-09 04:28:29 +00:00
// we may fail to create a RulesetContainer if the beatmap cannot be loaded with the user's preferred ruleset
2017-04-20 02:36:50 +00:00
// let's try again forcing the beatmap's ruleset.
ruleset = beatmap.BeatmapInfo.Ruleset;
2017-04-20 02:36:50 +00:00
rulesetInstance = ruleset.CreateInstance();
2017-08-09 04:28:29 +00:00
RulesetContainer = rulesetInstance.CreateRulesetContainerWith(Beatmap, true);
2017-04-17 06:44:46 +00:00
}
2017-04-26 11:22:03 +00:00
2017-08-09 04:28:29 +00:00
if (!RulesetContainer.Objects.Any())
2017-05-06 16:38:17 +00:00
throw new InvalidOperationException("Beatmap contains no hit objects!");
}
catch (Exception e)
{
Logger.Error(e, "Could not load beatmap sucessfully!");
//couldn't load, hard abort!
Exit();
return;
}
2016-10-05 11:49:31 +00:00
2018-02-27 07:05:21 +00:00
sourceClock = (IAdjustableClock)working.Track ?? new StopwatchClock();
adjustableClock = new DecoupleableInterpolatingFramedClock { IsCoupled = false };
2017-08-09 04:28:29 +00:00
var firstObjectTime = RulesetContainer.Objects.First().StartTime;
adjustableClock.Seek(AllowLeadIn
? Math.Min(0, firstObjectTime - Math.Max(beatmap.ControlPointInfo.TimingPointAt(firstObjectTime).BeatLength * 4, beatmap.BeatmapInfo.AudioLeadIn))
: firstObjectTime);
adjustableClock.ProcessFrame();
2017-04-26 09:07:22 +00:00
// the final usable gameplay clock with user-set offsets applied.
var offsetClock = new FramedOffsetClock(adjustableClock);
userAudioOffset.ValueChanged += v => offsetClock.Offset = v;
userAudioOffset.TriggerChange();
scoreProcessor = RulesetContainer.CreateScoreProcessor();
Children = new Drawable[]
2016-10-06 14:33:09 +00:00
{
pauseContainer = new PauseContainer(offsetClock, adjustableClock)
2017-09-14 12:28:53 +00:00
{
OnRetry = Restart,
OnQuit = Exit,
CheckCanPause = () => AllowPause && ValidForResume && !HasFailed && !RulesetContainer.HasReplayLoaded,
2018-01-21 15:11:43 +00:00
OnPause = () =>
{
pauseContainer.Retries = RestartCount;
hudOverlay.KeyCounter.IsCounting = pauseContainer.IsPaused;
},
2018-01-24 08:48:42 +00:00
OnResume = () => hudOverlay.KeyCounter.IsCounting = true,
Children = new Drawable[]
{
storyboardContainer = new Container
2017-02-28 11:14:48 +00:00
{
2017-04-26 09:07:22 +00:00
RelativeSizeAxes = Axes.Both,
Alpha = 0,
2017-02-28 11:14:48 +00:00
},
2018-03-20 07:26:36 +00:00
new LocalSkinOverrideContainer(working.Skin)
{
RelativeSizeAxes = Axes.Both,
Child = RulesetContainer
},
2018-03-05 15:10:53 +00:00
new SkipOverlay(firstObjectTime)
{
Clock = Clock, // skip button doesn't want to use the audio clock directly
ProcessCustomClock = false,
AdjustableClock = adjustableClock,
FramedClock = offsetClock,
},
hudOverlay = new HUDOverlay(scoreProcessor, RulesetContainer, working, offsetClock, adjustableClock)
{
Clock = Clock, // hud overlay doesn't want to use the audio clock directly
ProcessCustomClock = false,
Anchor = Anchor.Centre,
Origin = Anchor.Centre
},
2018-01-12 20:59:36 +00:00
new BreakOverlay(beatmap.BeatmapInfo.LetterboxInBreaks, scoreProcessor)
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
2018-02-28 12:50:52 +00:00
ProcessCustomClock = false,
Breaks = beatmap.Breaks
2018-01-12 10:50:09 +00:00
}
}
},
2017-04-06 06:34:52 +00:00
failOverlay = new FailOverlay
{
OnRetry = Restart,
OnQuit = Exit,
2017-04-09 13:26:31 +00:00
},
2017-04-10 03:06:10 +00:00
new HotkeyRetryOverlay
2017-04-09 13:26:31 +00:00
{
2018-01-21 15:11:43 +00:00
Action = () =>
{
2018-01-22 06:06:27 +00:00
if (!IsCurrentScreen) return;
//we want to hide the hitrenderer immediately (looks better).
//we may be able to remove this once the mouse cursor trail is improved.
RulesetContainer?.Hide();
Restart();
},
2017-04-06 06:34:52 +00:00
}
};
2018-02-08 20:28:42 +00:00
if (ShowStoryboard)
initializeStoryboard(false);
2017-09-14 12:28:53 +00:00
// Bind ScoreProcessor to ourselves
scoreProcessor.AllJudged += onCompletion;
scoreProcessor.Failed += onFail;
foreach (var mod in Beatmap.Value.Mods.Value.OfType<IApplicableToScoreProcessor>())
mod.ApplyToScoreProcessor(scoreProcessor);
}
private void applyRateFromMods()
{
2018-02-27 07:05:21 +00:00
if (sourceClock == null) return;
2018-02-27 07:05:21 +00:00
sourceClock.Rate = 1;
foreach (var mod in Beatmap.Value.Mods.Value.OfType<IApplicableToClock>())
2018-02-27 07:05:21 +00:00
mod.ApplyToClock(sourceClock);
}
public void Restart()
{
sampleRestart?.Play();
2017-04-18 06:52:38 +00:00
ValidForResume = false;
RestartRequested?.Invoke();
Exit();
}
private ScheduledDelegate onCompletionEvent;
private void onCompletion()
2016-11-29 14:59:56 +00:00
{
// Only show the completion screen if the player hasn't failed
if (scoreProcessor.HasFailed || onCompletionEvent != null)
return;
ValidForResume = false;
if (!AllowResults) return;
using (BeginDelayedSequence(1000))
2016-11-29 14:59:56 +00:00
{
onCompletionEvent = Schedule(delegate
{
2018-01-22 06:06:27 +00:00
if (!IsCurrentScreen) return;
var score = new Score
{
2018-01-22 06:06:27 +00:00
Beatmap = Beatmap.Value.BeatmapInfo,
Ruleset = ruleset
};
scoreProcessor.PopulateScore(score);
score.User = RulesetContainer.Replay?.User ?? api.LocalUser.Value;
Push(new Results(score));
});
}
2016-11-29 14:59:56 +00:00
}
2017-08-05 02:59:58 +00:00
private bool onFail()
2017-01-20 07:51:43 +00:00
{
if (Beatmap.Value.Mods.Value.OfType<IApplicableFailOverride>().Any(m => !m.AllowFail))
2017-08-05 02:59:58 +00:00
return false;
adjustableClock.Stop();
2017-01-20 07:51:43 +00:00
2017-04-01 18:17:24 +00:00
HasFailed = true;
failOverlay.Retries = RestartCount;
failOverlay.Show();
2017-08-05 02:59:58 +00:00
return true;
2017-01-20 07:51:43 +00:00
}
2017-02-17 09:59:30 +00:00
protected override void OnEntering(Screen last)
{
base.OnEntering(last);
2017-07-19 10:10:04 +00:00
if (!loadedSuccessfully)
return;
2017-09-14 12:28:53 +00:00
Content.Alpha = 0;
2017-07-16 15:28:20 +00:00
Content
.ScaleTo(0.7f)
2017-07-22 18:50:25 +00:00
.ScaleTo(1, 750, Easing.OutQuint)
2017-07-16 15:28:20 +00:00
.Delay(250)
.FadeIn(250);
Task.Run(() =>
{
2018-02-27 07:05:21 +00:00
sourceClock.Reset();
Schedule(() =>
{
adjustableClock.ChangeSource(sourceClock);
applyRateFromMods();
this.Delay(750).Schedule(() =>
{
if (!pauseContainer.IsPaused)
2018-02-27 07:02:57 +00:00
{
adjustableClock.Start();
2018-02-27 07:02:57 +00:00
}
});
});
});
pauseContainer.Alpha = 0;
2017-07-22 18:50:25 +00:00
pauseContainer.FadeIn(750, Easing.OutQuint);
}
protected override void OnSuspending(Screen next)
{
fadeOut();
base.OnSuspending(next);
}
2016-11-16 06:48:35 +00:00
2017-02-17 09:59:30 +00:00
protected override bool OnExiting(Screen next)
2016-12-16 16:13:24 +00:00
{
if ((!AllowPause || HasFailed || !ValidForResume || pauseContainer?.IsPaused != false || RulesetContainer?.HasReplayLoaded != false) && (!pauseContainer?.IsResuming ?? true))
{
// In the case of replays, we may have changed the playback rate.
applyRateFromMods();
fadeOut();
return base.OnExiting(next);
}
2017-07-19 10:10:04 +00:00
if (loadedSuccessfully)
2018-02-09 19:04:39 +00:00
pauseContainer?.Pause();
2017-07-19 10:10:04 +00:00
return true;
2016-12-16 16:13:24 +00:00
}
private void fadeOut()
2017-09-14 12:28:53 +00:00
{
const float fade_out_duration = 250;
2017-08-09 04:28:29 +00:00
RulesetContainer?.FadeOut(fade_out_duration);
Content.FadeOut(fade_out_duration);
2017-07-22 18:50:25 +00:00
hudOverlay?.ScaleTo(0.7f, fade_out_duration * 3, Easing.In);
2018-01-09 13:21:15 +00:00
Background?.FadeTo(1f, fade_out_duration);
2016-12-16 16:13:24 +00:00
}
2017-09-14 12:28:53 +00:00
protected override bool OnWheel(InputState state) => mouseWheelDisabled.Value && !pauseContainer.IsPaused;
private void initializeStoryboard(bool asyncLoad)
{
if (storyboardContainer == null)
return;
var beatmap = Beatmap.Value;
2018-02-16 03:08:21 +00:00
storyboard = beatmap.Storyboard.CreateDrawable();
storyboard.Masking = true;
if (asyncLoad)
LoadComponentAsync(storyboard, storyboardContainer.Add);
else
storyboardContainer.Add(storyboard);
2017-09-14 12:28:53 +00:00
}
protected override void UpdateBackgroundElements()
{
if (!IsCurrentScreen) return;
base.UpdateBackgroundElements();
if (ShowStoryboard && storyboard == null)
initializeStoryboard(true);
var beatmap = Beatmap.Value;
var storyboardVisible = ShowStoryboard && beatmap.Storyboard.HasDrawable;
2016-12-16 16:13:24 +00:00
storyboardContainer?
2018-02-16 03:08:21 +00:00
.FadeColour(OsuColour.Gray(BackgroundOpacity), BACKGROUND_FADE_DURATION, Easing.OutQuint)
.FadeTo(storyboardVisible && BackgroundOpacity > 0 ? 1 : 0, BACKGROUND_FADE_DURATION, Easing.OutQuint);
2018-02-16 03:08:21 +00:00
if (storyboardVisible && beatmap.Storyboard.ReplacesBackground)
Background?.FadeTo(0, BACKGROUND_FADE_DURATION, Easing.OutQuint);
}
2016-09-29 11:13:58 +00:00
}
2017-04-11 15:09:45 +00:00
}