osu/osu.Game/Screens/Play/ReplayPlayer.cs

98 lines
3.3 KiB
C#
Raw Normal View History

// 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
2021-05-25 09:37:04 +00:00
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using osu.Framework.Input.Bindings;
using osu.Framework.Threading;
using osu.Game.Beatmaps;
using osu.Game.Extensions;
using osu.Game.Input.Bindings;
2021-05-25 09:37:04 +00:00
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Objects;
2018-11-29 04:22:45 +00:00
using osu.Game.Scoring;
using osu.Game.Screens.Ranking;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Screens.Play
{
public class ReplayPlayer : Player, IKeyBindingHandler<GlobalAction>
2018-04-13 09:19:50 +00:00
{
private readonly Func<IBeatmap, IReadOnlyList<Mod>, Score> createScore;
2018-04-13 09:19:50 +00:00
2019-09-19 05:00:41 +00:00
// Disallow replays from failing. (see https://github.com/ppy/osu/issues/6108)
protected override bool CheckModsAllowFailure() => false;
2019-09-18 19:49:48 +00:00
public ReplayPlayer(Score score, PlayerConfiguration configuration = null)
2021-05-25 09:37:04 +00:00
: this((_, __) => score, configuration)
{
}
public ReplayPlayer(Func<IBeatmap, IReadOnlyList<Mod>, Score> createScore, PlayerConfiguration configuration = null)
: base(configuration)
2018-04-13 09:19:50 +00:00
{
2021-05-25 09:37:04 +00:00
this.createScore = createScore;
}
2020-03-23 10:31:43 +00:00
protected override void PrepareReplay()
2018-04-13 09:19:50 +00:00
{
DrawableRuleset?.SetReplayScore(Score);
2020-06-19 12:54:09 +00:00
}
protected override Score CreateScore() => createScore(GameplayBeatmap.PlayableBeatmap, Mods.Value);
// Don't re-import replay scores as they're already present in the database.
protected override Task ImportScore(Score score) => Task.CompletedTask;
protected override ResultsScreen CreateResults(ScoreInfo score) => new SoloResultsScreen(score, false);
private ScheduledDelegate keyboardSeekDelegate;
public bool OnPressed(GlobalAction action)
{
const double keyboard_seek_amount = 5000;
switch (action)
{
case GlobalAction.SeekReplayBackward:
keyboardSeekDelegate?.Cancel();
keyboardSeekDelegate = this.BeginKeyRepeat(Scheduler, () => keyboardSeek(-1));
return true;
case GlobalAction.SeekReplayForward:
keyboardSeekDelegate?.Cancel();
keyboardSeekDelegate = this.BeginKeyRepeat(Scheduler, () => keyboardSeek(1));
return true;
case GlobalAction.TogglePauseReplay:
if (GameplayClockContainer.IsPaused.Value)
GameplayClockContainer.Start();
else
GameplayClockContainer.Stop();
return true;
}
return false;
void keyboardSeek(int direction)
{
double target = Math.Clamp(GameplayClockContainer.CurrentTime + direction * keyboard_seek_amount, 0, GameplayBeatmap.HitObjects.Last().GetEndTime());
Seek(target);
}
}
public void OnReleased(GlobalAction action)
{
switch (action)
{
case GlobalAction.SeekReplayBackward:
case GlobalAction.SeekReplayForward:
keyboardSeekDelegate?.Cancel();
break;
}
}
2018-04-13 09:19:50 +00:00
}
}