2017-02-07 12:59:30 +08:00
|
|
|
|
// 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
|
|
|
|
{
|
2017-03-23 12:41:50 +08:00
|
|
|
|
private readonly ManualClock clock = new ManualClock();
|
2017-02-28 19:14:48 +08:00
|
|
|
|
private IFrameBasedClock parentClock;
|
|
|
|
|
|
2017-03-04 14:29:15 +08:00
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-25 18:09:30 +08:00
|
|
|
|
/// <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)
|
2017-04-25 18:09:30 +08:00
|
|
|
|
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;
|
|
|
|
|
|
2017-04-25 18:09:30 +08:00
|
|
|
|
if (!isAttached)
|
2017-02-28 19:14:48 +08:00
|
|
|
|
{
|
|
|
|
|
clock.CurrentTime = parentClock.CurrentTime;
|
|
|
|
|
}
|
2017-04-25 18:09:30 +08:00
|
|
|
|
else
|
2017-02-28 19:14:48 +08:00
|
|
|
|
{
|
|
|
|
|
double? newTime = replayInputHandler.SetFrameFromTime(parentClock.CurrentTime);
|
|
|
|
|
|
|
|
|
|
if (newTime == null)
|
2017-04-25 18:09:30 +08:00
|
|
|
|
{
|
|
|
|
|
// 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;
|
|
|
|
|
}
|
2017-04-25 18:09:30 +08:00
|
|
|
|
|
|
|
|
|
requireMoreUpdateLoops = clock.CurrentTime != parentClock.CurrentTime;
|
|
|
|
|
base.Update();
|
2017-02-28 19:14:48 +08:00
|
|
|
|
}
|
2017-01-16 11:40:52 +08:00
|
|
|
|
}
|
|
|
|
|
}
|