1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-10 01:12:56 +08:00
osu-lazer/osu.Game/Screens/Play/Player.cs

401 lines
14 KiB
C#
Raw Normal View History

// 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
using OpenTK;
2016-11-09 07:13:20 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2016-11-14 16:23:33 +08:00
using osu.Framework.Audio.Track;
using osu.Framework.Configuration;
2016-11-14 16:23:33 +08:00
using osu.Framework.Graphics;
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;
using osu.Game.Configuration;
2016-11-14 16:23:33 +08:00
using osu.Game.Database;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets;
using osu.Game.Rulesets.UI;
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;
using osu.Framework.Threading;
using osu.Game.Rulesets.Mods;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Screens.Ranking;
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
internal override bool ShowOverlays => false;
internal override bool HasLocalCursorDisplayed => !IsPaused && !HasFailed && HitRenderer.ProvidingUserCursor;
2017-03-16 22:58:36 +08:00
2016-10-27 16:53:37 +08:00
public BeatmapInfo BeatmapInfo;
public Action RestartRequested;
public bool IsPaused => !decoupledClock.IsRunning;
internal override bool AllowRulesetChange => false;
2017-04-02 02:17:24 +08:00
public bool HasFailed { get; private set; }
public int RestartCount;
2017-03-23 12:52:38 +08:00
private const double pause_cooldown = 1000;
2017-03-07 09:59:19 +08:00
private double lastPauseActionTime;
private bool canPause => ValidForResume && !HasFailed && Time.Current >= lastPauseActionTime + pause_cooldown;
private IAdjustableClock adjustableSourceClock;
private FramedOffsetClock offsetClock;
private DecoupleableInterpolatingFramedClock decoupledClock;
2016-11-16 14:48:35 +08:00
private RulesetInfo ruleset;
2016-11-29 19:30:16 +08:00
private ScoreProcessor scoreProcessor;
protected HitRenderer HitRenderer;
#region User Settings
private Bindable<double> dimLevel;
private Bindable<bool> mouseWheelDisabled;
private Bindable<double> userAudioOffset;
#endregion
2017-01-27 20:57:22 +08:00
private SkipButton skipButton;
2016-11-29 19:30:16 +08:00
2017-04-26 17:07:22 +08:00
private Container hitRendererContainer;
2017-03-10 12:02:50 +08:00
private HudOverlay hudOverlay;
private PauseOverlay pauseOverlay;
2017-03-28 15:49:58 +08:00
private FailOverlay failOverlay;
2017-04-17 14:44:46 +08:00
[BackgroundDependencyLoader(permitNulls: true)]
private void load(AudioManager audio, BeatmapDatabase beatmaps, OsuConfigManager config, OsuGame osu)
2016-10-05 19:49:31 +08:00
{
dimLevel = config.GetBindable<double>(OsuConfig.DimLevel);
2017-02-28 18:44:12 +08:00
mouseWheelDisabled = config.GetBindable<bool>(OsuConfig.MouseDisableWheel);
Ruleset rulesetInstance;
try
{
if (Beatmap == null)
Beatmap = beatmaps.GetWorkingBeatmap(BeatmapInfo, withStoryboard: true);
2017-04-26 17:32:40 +08:00
if (Beatmap?.Beatmap == null)
throw new Exception("Beatmap was not loaded");
2017-04-17 14:44:46 +08:00
2017-04-20 10:36:50 +08:00
ruleset = osu?.Ruleset.Value ?? Beatmap.BeatmapInfo.Ruleset;
rulesetInstance = ruleset.CreateInstance();
2017-04-17 14:44:46 +08:00
try
{
2017-04-20 10:36:50 +08:00
HitRenderer = rulesetInstance.CreateHitRendererWith(Beatmap);
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-04-20 10:36:50 +08:00
// we may fail to create a HitRenderer if the beatmap cannot be loaded with the user's preferred ruleset
// let's try again forcing the beatmap's ruleset.
ruleset = Beatmap.BeatmapInfo.Ruleset;
2017-04-20 10:36:50 +08:00
rulesetInstance = ruleset.CreateInstance();
HitRenderer = rulesetInstance.CreateHitRendererWith(Beatmap);
2017-04-17 14:44:46 +08:00
}
2017-04-26 19:22:03 +08:00
if (!HitRenderer.Objects.Any())
throw new Exception("Beatmap contains no hit objects!");
}
catch (Exception e)
{
Logger.Log($"Could not load this beatmap sucessfully ({e})!", LoggingTarget.Runtime, LogLevel.Error);
//couldn't load, hard abort!
Exit();
return;
}
2016-10-05 19:49:31 +08:00
Track track = Beatmap.Track;
if (track != null)
{
2016-11-09 07:13:20 +08:00
audio.Track.SetExclusive(track);
adjustableSourceClock = track;
}
adjustableSourceClock = (IAdjustableClock)track ?? new StopwatchClock();
2017-04-26 17:32:47 +08:00
decoupledClock = new DecoupleableInterpolatingFramedClock { IsCoupled = false };
2017-04-26 17:07:22 +08:00
var firstObjectTime = HitRenderer.Objects.First().StartTime;
decoupledClock.Seek(Math.Min(0, firstObjectTime - Math.Max(Beatmap.Beatmap.TimingInfo.BeatLengthAt(firstObjectTime) * 4, Beatmap.BeatmapInfo.AudioLeadIn)));
decoupledClock.ProcessFrame();
offsetClock = new FramedOffsetClock(decoupledClock);
userAudioOffset = config.GetBindable<double>(OsuConfig.AudioOffset);
userAudioOffset.ValueChanged += v => offsetClock.Offset = v;
userAudioOffset.TriggerChange();
Schedule(() =>
{
adjustableSourceClock.Reset();
foreach (var mod in Beatmap.Mods.Value.OfType<IApplicableToClock>())
mod.ApplyToClock(adjustableSourceClock);
2017-04-26 17:07:22 +08:00
decoupledClock.ChangeSource(adjustableSourceClock);
});
scoreProcessor = HitRenderer.CreateScoreProcessor();
hudOverlay = new StandardHudOverlay()
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre
};
hudOverlay.KeyCounter.Add(rulesetInstance.CreateGameplayKeys());
hudOverlay.BindProcessor(scoreProcessor);
hudOverlay.BindHitRenderer(HitRenderer);
hudOverlay.Progress.Objects = HitRenderer.Objects;
hudOverlay.Progress.AudioClock = decoupledClock;
hudOverlay.Progress.AllowSeeking = HitRenderer.HasReplayLoaded;
2017-04-26 19:22:03 +08:00
hudOverlay.Progress.OnSeek = pos => decoupledClock.Seek(pos);
2017-01-20 15:51:43 +08:00
//bind HitRenderer to ScoreProcessor and ourselves (for a pass situation)
HitRenderer.OnAllJudged += onCompletion;
2017-01-20 15:51:43 +08:00
//bind ScoreProcessor to ourselves (for a fail situation)
scoreProcessor.Failed += onFail;
Children = new Drawable[]
2016-10-06 22:33:09 +08:00
{
2017-04-26 17:07:22 +08:00
hitRendererContainer = new Container
{
2017-02-28 19:14:48 +08:00
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
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,
Children = new Drawable[]
{
HitRenderer,
skipButton = new SkipButton { Alpha = 0 },
}
2017-02-28 19:14:48 +08:00
},
}
},
hudOverlay,
2017-04-06 14:34:52 +08:00
pauseOverlay = new PauseOverlay
{
OnResume = delegate
{
Delay(400);
Schedule(Resume);
},
OnRetry = Restart,
OnQuit = Exit,
},
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
{
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-04-11 23:09:45 +08:00
HitRenderer?.Hide();
Restart();
},
2017-04-06 14:34:52 +08:00
}
};
2016-10-05 19:49:31 +08:00
}
2017-01-27 20:57:22 +08:00
private void initializeSkipButton()
{
const double skip_required_cutoff = 3000;
const double fade_time = 300;
double firstHitObject = Beatmap.Beatmap.HitObjects.First().StartTime;
if (firstHitObject < skip_required_cutoff)
{
skipButton.Alpha = 0;
skipButton.Expire();
return;
}
skipButton.FadeInFromZero(fade_time);
skipButton.Action = () =>
{
decoupledClock.Seek(firstHitObject - skip_required_cutoff - fade_time);
2017-01-27 20:57:22 +08:00
skipButton.Action = null;
};
skipButton.Delay(firstHitObject - skip_required_cutoff - fade_time);
skipButton.FadeOut(fade_time);
skipButton.Expire();
}
public void Pause(bool force = false)
{
2017-04-12 19:01:52 +08:00
if (!canPause && !force) return;
// the actual pausing is potentially happening on a different thread.
// we want to wait for the source clock to stop so we can be sure all components are in a stable state.
if (!IsPaused)
{
decoupledClock.Stop();
Schedule(() => Pause(force));
return;
}
// we need to do a final check after all of our children have processed up to the paused clock time.
// this is to cover cases where, for instance, the player fails in the last processed frame (which would change canPause).
// as the scheduler runs before children updates, let's schedule for the next frame.
Schedule(() =>
{
if (!canPause) return;
lastPauseActionTime = Time.Current;
hudOverlay.KeyCounter.IsCounting = false;
hudOverlay.Progress.Show();
2017-01-31 21:17:47 +08:00
pauseOverlay.Retries = RestartCount;
pauseOverlay.Show();
});
}
public void Resume()
{
lastPauseActionTime = Time.Current;
hudOverlay.KeyCounter.IsCounting = true;
hudOverlay.Progress.Hide();
pauseOverlay.Hide();
decoupledClock.Start();
}
public void Restart()
{
2017-04-18 14:52:38 +08:00
ValidForResume = false;
RestartRequested?.Invoke();
Exit();
}
private ScheduledDelegate onCompletionEvent;
private void onCompletion()
2016-11-29 22:59:56 +08:00
{
// Only show the completion screen if the player hasn't failed
if (scoreProcessor.HasFailed || onCompletionEvent != null)
return;
ValidForResume = false;
2016-11-29 22:59:56 +08:00
Delay(1000);
onCompletionEvent = Schedule(delegate
2016-11-29 22:59:56 +08:00
{
var score = new Score
{
Beatmap = Beatmap.BeatmapInfo,
Ruleset = ruleset
};
scoreProcessor.PopulateScore(score);
score.User = HitRenderer.Replay?.User ?? (Game as OsuGame)?.API?.LocalUser?.Value;
Push(new Results(score));
2016-11-29 22:59:56 +08:00
});
}
2017-01-20 15:51:43 +08:00
private void onFail()
{
decoupledClock.Stop();
2017-01-20 15:51:43 +08:00
2017-04-02 02:17:24 +08:00
HasFailed = true;
failOverlay.Retries = RestartCount;
failOverlay.Show();
2017-01-20 15:51:43 +08:00
}
2017-02-17 17:59:30 +08:00
protected override void OnEntering(Screen last)
{
base.OnEntering(last);
2017-02-22 20:43:29 +08:00
(Background as BackgroundScreenBeatmap)?.BlurTo(Vector2.Zero, 1500, EasingTypes.OutQuint);
Background?.FadeTo(1 - (float)dimLevel, 1500, EasingTypes.OutQuint);
Content.Alpha = 0;
dimLevel.ValueChanged += newDim => Background?.FadeTo(1 - (float)newDim, 800);
2017-02-22 20:43:29 +08:00
Content.ScaleTo(0.7f);
Content.Delay(250);
Content.FadeIn(250);
2017-02-22 20:43:29 +08:00
Content.ScaleTo(1, 750, EasingTypes.OutQuint);
Delay(750);
Schedule(() =>
{
decoupledClock.Start();
initializeSkipButton();
});
2017-04-26 17:07:22 +08:00
hitRendererContainer.Alpha = 0;
hitRendererContainer.FadeIn(750, EasingTypes.OutQuint);
}
protected override void OnSuspending(Screen next)
{
fadeOut();
base.OnSuspending(next);
}
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
{
if (!HasFailed && ValidForResume)
{
if (pauseOverlay != null && !HitRenderer.HasReplayLoaded)
{
//pause screen override logic.
if (pauseOverlay?.State == Visibility.Hidden && !canPause) return true;
if (!IsPaused) // For if the user presses escape quickly when entering the map
{
Pause();
return true;
}
}
}
fadeOut();
return base.OnExiting(next);
2016-12-17 00:13:24 +08:00
}
private void fadeOut()
{
const float fade_out_duration = 250;
HitRenderer?.FadeOut(fade_out_duration);
Content.FadeOut(fade_out_duration);
hudOverlay.ScaleTo(0.7f, fade_out_duration * 3, EasingTypes.In);
Background?.FadeTo(1f, fade_out_duration);
2016-12-17 00:13:24 +08:00
}
2017-03-07 09:59:19 +08:00
protected override bool OnWheel(InputState state) => mouseWheelDisabled.Value && !IsPaused;
2016-09-29 19:13:58 +08:00
}
2017-04-11 23:09:45 +08:00
}