2019-01-24 08:43:03 +00:00
|
|
|
|
// 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
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2018-09-18 08:05:26 +00:00
|
|
|
|
using System.Linq;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Framework.Input.Handlers;
|
2018-07-21 02:38:28 +00:00
|
|
|
|
using osu.Framework.Input.StateChanges;
|
2018-09-18 08:05:26 +00:00
|
|
|
|
using osu.Framework.Input.StateChanges.Events;
|
2018-07-21 02:38:28 +00:00
|
|
|
|
using osu.Framework.Input.States;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
using osu.Framework.Platform;
|
2018-09-18 08:05:26 +00:00
|
|
|
|
using osu.Game.Rulesets.UI;
|
2018-11-20 07:51:59 +00:00
|
|
|
|
using osuTK;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Input.Handlers
|
|
|
|
|
{
|
|
|
|
|
public abstract class ReplayInputHandler : InputHandler
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// A function that converts coordinates from gamefield to screen space.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Func<Vector2, Vector2> GamefieldToScreenSpace { protected get; set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Update the current frame based on an incoming time value.
|
|
|
|
|
/// There are cases where we return a "must-use" time value that is different from the input.
|
|
|
|
|
/// This is to ensure accurate playback of replay data.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="time">The time which we should use for finding the current frame.</param>
|
|
|
|
|
/// <returns>The usable time value. If null, we should not advance time as we do not have enough data.</returns>
|
|
|
|
|
public abstract double? SetFrameFromTime(double time);
|
|
|
|
|
|
|
|
|
|
public override bool Initialize(GameHost host) => true;
|
|
|
|
|
|
|
|
|
|
public override bool IsActive => true;
|
|
|
|
|
|
|
|
|
|
public override int Priority => 0;
|
|
|
|
|
|
2018-06-11 14:00:26 +00:00
|
|
|
|
public class ReplayState<T> : IInput
|
2018-04-13 09:19:50 +00:00
|
|
|
|
where T : struct
|
|
|
|
|
{
|
|
|
|
|
public List<T> PressedActions;
|
|
|
|
|
|
2018-06-11 14:00:26 +00:00
|
|
|
|
public void Apply(InputState state, IInputStateChangeHandler handler)
|
2018-04-13 09:19:50 +00:00
|
|
|
|
{
|
2018-09-18 08:05:26 +00:00
|
|
|
|
if (!(state is RulesetInputManagerInputState<T> inputState))
|
|
|
|
|
throw new InvalidOperationException($"{nameof(ReplayState<T>)} should only be applied to a {nameof(RulesetInputManagerInputState<T>)}");
|
|
|
|
|
|
|
|
|
|
var lastPressed = inputState.LastReplayState?.PressedActions ?? new List<T>();
|
|
|
|
|
var released = lastPressed.Except(PressedActions).ToArray();
|
|
|
|
|
var pressed = PressedActions.Except(lastPressed).ToArray();
|
|
|
|
|
|
|
|
|
|
inputState.LastReplayState = this;
|
|
|
|
|
|
|
|
|
|
handler.HandleInputStateChange(new ReplayStateChangeEvent<T>(state, this, released, pressed));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class ReplayStateChangeEvent<T> : InputStateChangeEvent
|
|
|
|
|
{
|
|
|
|
|
public readonly T[] ReleasedActions;
|
|
|
|
|
public readonly T[] PressedActions;
|
|
|
|
|
|
|
|
|
|
public ReplayStateChangeEvent(InputState state, IInput input, T[] releasedActions, T[] pressedActions)
|
|
|
|
|
: base(state, input)
|
|
|
|
|
{
|
|
|
|
|
ReleasedActions = releasedActions;
|
|
|
|
|
PressedActions = pressedActions;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|