1
0
mirror of https://github.com/ppy/osu.git synced 2024-10-01 04:17:51 +08:00
osu-lazer/osu.Game/Screens/Play/PlayerInputManager.cs

101 lines
3.2 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
2017-01-16 11:40:52 +08:00
using osu.Framework.Input;
2017-02-28 19:14:48 +08:00
using osu.Framework.Timing;
using osu.Game.Input.Handlers;
2017-01-16 11:40:52 +08:00
namespace osu.Game.Screens.Play
{
2017-02-28 19:14:48 +08:00
public class PlayerInputManager : PassThroughInputManager
2017-01-16 11:40:52 +08:00
{
private readonly ManualClock clock = new ManualClock();
2017-02-28 19:14:48 +08:00
private IFrameBasedClock parentClock;
private ReplayInputHandler replayInputHandler;
public ReplayInputHandler ReplayInputHandler
2017-01-16 11:40:52 +08:00
{
2017-02-28 19:14:48 +08:00
get { return replayInputHandler; }
set
{
if (replayInputHandler != null) RemoveHandler(replayInputHandler);
replayInputHandler = value;
UseParentState = replayInputHandler == null;
if (replayInputHandler != null)
AddHandler(replayInputHandler);
}
2017-01-16 11:40:52 +08:00
}
2017-02-28 19:14:48 +08:00
protected override void LoadComplete()
{
base.LoadComplete();
parentClock = Clock;
Clock = new FramedClock(clock);
}
/// <summary>
/// Whether we running up-to-date with our parent clock.
/// If not, we will need to keep processing children until we catch up.
/// </summary>
private bool requireMoreUpdateLoops;
/// <summary>
/// Whether we in a valid state (ie. should we keep processing children frames).
/// This should be set to false when the replay is, for instance, waiting for future frames to arrive.
/// </summary>
private bool validState;
protected override bool RequiresChildrenUpdate => base.RequiresChildrenUpdate && validState;
private bool isAttached => replayInputHandler != null && !UseParentState;
private const int max_catch_up_updates_per_frame = 50;
public override bool UpdateSubTree()
{
requireMoreUpdateLoops = true;
validState = true;
int loops = 0;
2017-04-26 19:15:34 +08:00
while (validState && requireMoreUpdateLoops && loops++ < max_catch_up_updates_per_frame)
if (!base.UpdateSubTree())
return false;
return true;
}
2017-02-28 19:14:48 +08:00
protected override void Update()
{
if (parentClock == null) return;
clock.Rate = parentClock.Rate;
clock.IsRunning = parentClock.IsRunning;
if (!isAttached)
2017-02-28 19:14:48 +08:00
{
clock.CurrentTime = parentClock.CurrentTime;
}
else
2017-02-28 19:14:48 +08:00
{
double? newTime = replayInputHandler.SetFrameFromTime(parentClock.CurrentTime);
if (newTime == null)
{
// we shouldn't execute for this time value. probably waiting on more replay data.
validState = false;
return;
}
2017-02-28 19:14:48 +08:00
clock.CurrentTime = newTime.Value;
}
requireMoreUpdateLoops = clock.CurrentTime != parentClock.CurrentTime;
base.Update();
2017-02-28 19:14:48 +08:00
}
2017-01-16 11:40:52 +08:00
}
}