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-04-26 17:56:20 +08:00
|
|
|
|
using OpenTK.Input;
|
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Configuration;
|
2017-01-16 11:40:52 +08:00
|
|
|
|
using osu.Framework.Input;
|
2017-02-28 19:14:48 +08:00
|
|
|
|
using osu.Framework.Timing;
|
2017-04-26 17:56:20 +08:00
|
|
|
|
using osu.Game.Configuration;
|
2017-02-28 19:14:48 +08:00
|
|
|
|
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-04-27 16:38:00 +08:00
|
|
|
|
private ManualClock clock;
|
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-04-26 18:25:41 +08:00
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return replayInputHandler;
|
|
|
|
|
}
|
2017-02-28 19:14:48 +08:00
|
|
|
|
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-04-26 17:56:20 +08:00
|
|
|
|
private Bindable<bool> mouseDisabled;
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(OsuConfigManager config)
|
|
|
|
|
{
|
|
|
|
|
mouseDisabled = config.GetBindable<bool>(OsuConfig.MouseDisableButtons);
|
|
|
|
|
}
|
|
|
|
|
|
2017-02-28 19:14:48 +08:00
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
|
|
|
|
|
2017-04-27 16:38:00 +08:00
|
|
|
|
//our clock will now be our parent's clock, but we want to replace this to allow manual control.
|
2017-02-28 19:14:48 +08:00
|
|
|
|
parentClock = Clock;
|
2017-04-27 16:38:00 +08:00
|
|
|
|
|
|
|
|
|
Clock = new FramedClock(clock = new ManualClock
|
|
|
|
|
{
|
|
|
|
|
CurrentTime = parentClock.CurrentTime,
|
|
|
|
|
Rate = parentClock.Rate,
|
|
|
|
|
});
|
2017-02-28 19:14:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
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-04-26 17:56:20 +08:00
|
|
|
|
|
|
|
|
|
protected override void TransformState(InputState state)
|
|
|
|
|
{
|
|
|
|
|
base.TransformState(state);
|
|
|
|
|
|
|
|
|
|
// we don't want to transform the state if a replay is present (for now, at least).
|
|
|
|
|
if (replayInputHandler != null) return;
|
|
|
|
|
|
|
|
|
|
var mouse = state.Mouse as Framework.Input.MouseState;
|
|
|
|
|
|
|
|
|
|
if (mouse != null)
|
|
|
|
|
{
|
|
|
|
|
if (mouseDisabled.Value)
|
|
|
|
|
{
|
|
|
|
|
mouse.SetPressed(MouseButton.Left, false);
|
|
|
|
|
mouse.SetPressed(MouseButton.Right, false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-01-16 11:40:52 +08:00
|
|
|
|
}
|
|
|
|
|
}
|