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

349 lines
11 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 11:13:58 +00:00
using OpenTK;
2016-11-08 23:13:20 +00:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2016-11-14 08:23:33 +00:00
using osu.Framework.Audio.Track;
using osu.Framework.Configuration;
2016-11-14 08:23:33 +00: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 08:23:33 +00:00
using osu.Framework.Timing;
using osu.Game.Configuration;
2016-11-14 08:23:33 +00:00
using osu.Game.Database;
2016-11-14 09:03:20 +00:00
using osu.Game.Modes;
2016-11-29 14:59:56 +00:00
using osu.Game.Modes.UI;
using osu.Game.Screens.Backgrounds;
2016-11-29 06:41:48 +00:00
using osu.Game.Screens.Ranking;
2016-12-17 19:59:41 +00:00
using System;
2017-01-27 12:57:22 +00:00
using System.Linq;
using osu.Game.Modes.Scoring;
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
{
2017-02-17 09:59:30 +00:00
public class Player : OsuScreen
2016-09-29 11:13:58 +00:00
{
2017-02-17 09:59:30 +00:00
protected override BackgroundScreen CreateBackground() => new BackgroundScreenBeatmap(Beatmap);
2016-10-05 11:03:52 +00:00
internal override bool ShowOverlays => false;
internal override bool HasLocalCursorDisplayed => !hasReplayLoaded && !IsPaused && !IsFailed;
2017-03-16 14:58:36 +00:00
private bool hasReplayLoaded => HitRenderer.InputManager.ReplayInputHandler != null;
2017-03-16 14:58:36 +00:00
2016-10-27 08:53:37 +00:00
public BeatmapInfo BeatmapInfo;
2017-03-07 01:59:19 +00:00
public bool IsPaused { get; private set; }
public bool IsFailed { get; private set; }
public int RestartCount;
2017-03-23 04:52:38 +00:00
private const double pause_cooldown = 1000;
2017-03-07 01:59:19 +00:00
private double lastPauseActionTime;
2017-03-23 04:52:38 +00:00
private bool canPause => Time.Current >= lastPauseActionTime + pause_cooldown;
private IAdjustableClock sourceClock;
2017-02-28 11:14:48 +00:00
private IFrameBasedClock interpolatedSourceClock;
2016-11-16 06:48:35 +00:00
private Ruleset ruleset;
2016-11-29 11:30:16 +00:00
private ScoreProcessor scoreProcessor;
protected HitRenderer HitRenderer;
2016-12-18 09:48:59 +00:00
private Bindable<int> dimLevel;
2017-01-27 12:57:22 +00:00
private SkipButton skipButton;
2016-11-29 11:30:16 +00:00
2017-03-10 04:02:50 +00:00
private HudOverlay hudOverlay;
2017-03-28 07:49:58 +00:00
private PauseOverlay pauseOverlay;
private FailOverlay failOverlay;
[BackgroundDependencyLoader]
2017-03-28 08:41:08 +00:00
private void load(AudioManager audio, BeatmapDatabase beatmaps, OsuConfigManager config)
2016-10-05 11:49:31 +00:00
{
var beatmap = Beatmap.Beatmap;
if (beatmap.BeatmapInfo?.Mode > PlayMode.Osu)
{
//we only support osu! mode for now because the hitobject parsing is crappy and needs a refactor.
Exit();
return;
}
dimLevel = config.GetBindable<int>(OsuConfig.DimLevel);
2017-02-28 10:44:12 +00:00
mouseWheelDisabled = config.GetBindable<bool>(OsuConfig.MouseDisableWheel);
try
{
if (Beatmap == null)
Beatmap = beatmaps.GetWorkingBeatmap(BeatmapInfo, withStoryboard: true);
if ((Beatmap?.Beatmap?.HitObjects.Count ?? 0) == 0)
throw new Exception("No valid objects were found!");
if (Beatmap == null)
throw new Exception("Beatmap was not loaded");
}
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 11:49:31 +00:00
Track track = Beatmap.Track;
if (track != null)
{
2016-11-08 23:13:20 +00:00
audio.Track.SetExclusive(track);
sourceClock = track;
}
2016-10-28 10:55:48 +00:00
sourceClock = (IAdjustableClock)track ?? new StopwatchClock();
2017-02-28 11:14:48 +00:00
interpolatedSourceClock = new InterpolatingFramedClock(sourceClock);
Schedule(() =>
{
2016-10-28 10:55:48 +00:00
sourceClock.Reset();
});
ruleset = Ruleset.GetRuleset(Beatmap.PlayMode);
HitRenderer = ruleset.CreateHitRendererWith(Beatmap);
scoreProcessor = HitRenderer.CreateScoreProcessor();
hudOverlay = new StandardHudOverlay();
hudOverlay.KeyCounter.Add(ruleset.CreateGameplayKeys());
hudOverlay.BindProcessor(scoreProcessor);
2016-11-29 06:41:48 +00:00
2017-03-28 07:49:58 +00:00
pauseOverlay = new PauseOverlay
2017-01-30 13:06:26 +00:00
{
Depth = -1,
2017-03-28 07:49:58 +00:00
OnResume = delegate
{
Delay(400);
Schedule(Resume);
},
2017-03-28 07:49:58 +00:00
OnRetry = Restart,
OnQuit = Exit,
2017-03-28 03:08:31 +00:00
Title = @"paused",
Description = @"you're not going to do what i think you're going to do, are ya?",
2017-01-30 13:06:26 +00:00
};
2017-03-28 07:49:58 +00:00
failOverlay = new FailOverlay
{
Depth = -1,
2017-03-28 07:49:58 +00:00
OnRetry = Restart,
OnQuit = Exit,
2017-03-28 03:08:31 +00:00
Title = @"failed",
Description = @"you're dead, try again?",
};
hudOverlay.BindHitRenderer(HitRenderer);
2016-10-06 14:33:09 +00:00
2017-01-20 07:51:43 +00:00
//bind HitRenderer to ScoreProcessor and ourselves (for a pass situation)
HitRenderer.OnAllJudged += onCompletion;
2017-01-20 07:51:43 +00:00
//bind ScoreProcessor to ourselves (for a fail situation)
scoreProcessor.Failed += onFail;
Children = new Drawable[]
2016-10-06 14:33:09 +00:00
{
2017-02-28 11:14:48 +00:00
new Container
{
2017-02-28 11:14:48 +00:00
RelativeSizeAxes = Axes.Both,
Clock = interpolatedSourceClock,
Children = new Drawable[]
{
HitRenderer,
2017-02-28 11:14:48 +00:00
skipButton = new SkipButton
{
Alpha = 0
},
}
},
hudOverlay,
pauseOverlay,
failOverlay
};
2016-10-05 11:49:31 +00:00
}
2017-01-27 12:57:22 +00: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 = () =>
{
sourceClock.Seek(firstHitObject - skip_required_cutoff - fade_time);
skipButton.Action = null;
};
skipButton.Delay(firstHitObject - skip_required_cutoff - fade_time);
skipButton.FadeOut(fade_time);
skipButton.Expire();
}
public void Pause(bool force = false)
{
if (canPause || force)
{
lastPauseActionTime = Time.Current;
hudOverlay.KeyCounter.IsCounting = false;
2017-01-31 13:17:47 +00:00
pauseOverlay.Retries = RestartCount;
pauseOverlay.Show();
sourceClock.Stop();
2017-03-07 01:59:19 +00:00
IsPaused = true;
}
else
{
2017-03-07 01:59:19 +00:00
IsPaused = false;
}
}
public void Resume()
{
lastPauseActionTime = Time.Current;
hudOverlay.KeyCounter.IsCounting = true;
pauseOverlay.Hide();
sourceClock.Start();
2017-03-07 01:59:19 +00:00
IsPaused = false;
}
public void TogglePaused()
{
2017-03-07 01:59:19 +00:00
IsPaused = !IsPaused;
if (IsPaused) Pause(); else Resume();
}
public void Restart()
{
sourceClock.Stop(); // If the clock is running and Restart is called the game will lag until relaunch
var newPlayer = new Player();
newPlayer.LoadAsync(Game, delegate
{
newPlayer.RestartCount = RestartCount + 1;
ValidForResume = false;
if (!Push(newPlayer))
{
// Error(?)
}
});
}
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)
return;
2016-11-29 14:59:56 +00:00
Delay(1000);
Schedule(delegate
{
ValidForResume = false;
2016-11-29 14:59:56 +00:00
Push(new Results
{
2017-03-16 17:03:12 +00:00
Score = scoreProcessor.CreateScore()
2016-11-29 14:59:56 +00:00
});
});
}
2017-01-20 07:51:43 +00:00
private void onFail()
{
sourceClock.Stop();
Delay(500);
IsFailed = true;
failOverlay.Retries = RestartCount;
failOverlay.Show();
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-02-22 12:43:29 +00:00
(Background as BackgroundScreenBeatmap)?.BlurTo(Vector2.Zero, 1500, EasingTypes.OutQuint);
Background?.FadeTo((100f - dimLevel) / 100, 1500, EasingTypes.OutQuint);
Content.Alpha = 0;
2016-12-18 09:48:59 +00:00
dimLevel.ValueChanged += dimChanged;
2017-02-22 12:43:29 +00:00
Content.ScaleTo(0.7f);
Content.Delay(250);
Content.FadeIn(250);
2017-02-22 12:43:29 +00:00
Content.ScaleTo(1, 750, EasingTypes.OutQuint);
Delay(750);
Schedule(() =>
{
sourceClock.Start();
initializeSkipButton();
});
}
protected override void OnSuspending(Screen next)
{
Content.FadeOut(350);
2017-02-22 12:43:29 +00:00
Content.ScaleTo(0.7f, 750, EasingTypes.InQuint);
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 (pauseOverlay == null) return false;
2017-03-16 14:58:36 +00:00
if (hasReplayLoaded)
return false;
2017-02-01 02:33:28 +00:00
if (pauseOverlay.State != Visibility.Visible && !canPause) return true;
if (!IsPaused && sourceClock.IsRunning) // For if the user presses escape quickly when entering the map
{
Pause();
return true;
}
else
{
FadeOut(250);
Content.ScaleTo(0.7f, 750, EasingTypes.InQuint);
dimLevel.ValueChanged -= dimChanged;
Background?.FadeTo(1f, 200);
return base.OnExiting(next);
}
2016-12-16 16:13:24 +00:00
}
2016-12-17 19:59:41 +00:00
private void dimChanged(object sender, EventArgs e)
{
Background?.FadeTo((100f - dimLevel) / 100, 800);
}
2017-02-27 23:08:34 +00:00
private Bindable<bool> mouseWheelDisabled;
2017-03-07 01:59:19 +00:00
protected override bool OnWheel(InputState state) => mouseWheelDisabled.Value && !IsPaused;
2016-09-29 11:13:58 +00:00
}
}