osu/osu.Game.Tests/Visual/Gameplay/TestSceneReplayRecorder.cs

294 lines
11 KiB
C#
Raw Normal View History

2020-03-23 08:33:02 +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.
2021-10-01 17:22:23 +00:00
using System;
2020-03-23 08:33:02 +00:00
using System.Collections.Generic;
2020-03-24 07:22:54 +00:00
using System.Linq;
using NUnit.Framework;
using osu.Framework.Allocation;
2020-03-23 08:33:02 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Input.Bindings;
using osu.Framework.Input.Events;
2020-03-23 09:50:16 +00:00
using osu.Framework.Input.StateChanges;
2020-03-24 07:22:54 +00:00
using osu.Framework.Testing;
using osu.Framework.Threading;
using osu.Game.Beatmaps;
2020-03-23 09:50:16 +00:00
using osu.Game.Graphics.Sprites;
using osu.Game.Replays;
2020-03-23 08:33:02 +00:00
using osu.Game.Rulesets;
2021-10-01 17:22:23 +00:00
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Osu;
2020-03-23 09:50:16 +00:00
using osu.Game.Rulesets.Replays;
2020-03-23 08:33:02 +00:00
using osu.Game.Rulesets.UI;
using osu.Game.Scoring;
using osu.Game.Screens.Play;
2020-03-23 08:33:02 +00:00
using osu.Game.Tests.Visual.UserInterface;
using osuTK;
using osuTK.Graphics;
2020-03-24 07:22:54 +00:00
using osuTK.Input;
2020-03-23 08:33:02 +00:00
namespace osu.Game.Tests.Visual.Gameplay
2020-03-23 08:33:02 +00:00
{
2020-03-24 07:22:54 +00:00
public class TestSceneReplayRecorder : OsuManualInputManagerTestScene
2020-03-23 08:33:02 +00:00
{
2020-03-24 07:22:54 +00:00
private TestRulesetInputManager playbackManager;
private TestRulesetInputManager recordingManager;
2020-03-23 09:50:16 +00:00
2020-03-24 07:22:54 +00:00
private Replay replay;
private TestReplayRecorder recorder;
[Cached]
2021-10-01 17:22:23 +00:00
private GameplayState gameplayState = new GameplayState(new Beatmap(), new OsuRuleset(), Array.Empty<Mod>());
2020-03-24 07:22:54 +00:00
[SetUp]
public void SetUp() => Schedule(() =>
2020-03-23 08:33:02 +00:00
{
2020-03-24 07:22:54 +00:00
replay = new Replay();
2020-03-23 09:50:16 +00:00
Add(new GridContainer
2020-03-23 08:33:02 +00:00
{
2020-03-23 09:50:16 +00:00
RelativeSizeAxes = Axes.Both,
Content = new[]
2020-03-23 08:33:02 +00:00
{
2020-03-23 09:50:16 +00:00
new Drawable[]
{
2020-03-24 07:22:54 +00:00
recordingManager = new TestRulesetInputManager(new TestSceneModSettings.TestRulesetInfo(), 0, SimultaneousBindingMode.Unique)
2020-03-23 09:50:16 +00:00
{
2021-09-15 04:18:46 +00:00
Recorder = recorder = new TestReplayRecorder(new Score
{
Replay = replay,
2021-10-01 17:22:23 +00:00
ScoreInfo = { Beatmap = gameplayState.Beatmap.BeatmapInfo }
2021-09-15 04:18:46 +00:00
})
{
ScreenSpaceToGamefield = pos => recordingManager.ToLocalSpace(pos),
},
2020-03-23 09:50:16 +00:00
Child = new Container
{
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
new Box
{
Colour = Color4.Brown,
RelativeSizeAxes = Axes.Both,
},
new OsuSpriteText
{
Text = "Recording",
Scale = new Vector2(3),
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
2020-03-24 06:54:04 +00:00
new TestInputConsumer()
2020-03-23 09:50:16 +00:00
}
},
}
},
new Drawable[]
2020-03-23 08:33:02 +00:00
{
2020-03-23 09:50:16 +00:00
playbackManager = new TestRulesetInputManager(new TestSceneModSettings.TestRulesetInfo(), 0, SimultaneousBindingMode.Unique)
2020-03-23 08:33:02 +00:00
{
2020-03-23 09:50:16 +00:00
ReplayInputHandler = new TestFramedReplayInputHandler(replay)
{
GamefieldToScreenSpace = pos => playbackManager.ToScreenSpace(pos),
},
Child = new Container
{
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
new Box
{
Colour = Color4.DarkBlue,
RelativeSizeAxes = Axes.Both,
},
new OsuSpriteText
{
Text = "Playback",
Scale = new Vector2(3),
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
2020-03-24 06:54:04 +00:00
new TestInputConsumer()
2020-03-23 09:50:16 +00:00
}
},
}
2020-03-23 08:33:02 +00:00
}
2020-03-23 09:50:16 +00:00
}
2020-03-23 08:33:02 +00:00
});
2020-03-24 07:22:54 +00:00
});
[Test]
public void TestBasic()
{
AddStep("move to center", () => InputManager.MoveMouseTo(recordingManager.ScreenSpaceDrawQuad.Centre));
2021-04-19 08:02:59 +00:00
AddUntilStep("at least one frame recorded", () => replay.Frames.Count > 0);
AddUntilStep("position matches", () => playbackManager.ChildrenOfType<Box>().First().Position == recordingManager.ChildrenOfType<Box>().First().Position);
2020-03-24 07:22:54 +00:00
}
[Test]
public void TestHighFrameRate()
{
ScheduledDelegate moveFunction = null;
AddStep("move to center", () => InputManager.MoveMouseTo(recordingManager.ScreenSpaceDrawQuad.Centre));
AddStep("much move", () => moveFunction = Scheduler.AddDelayed(() =>
InputManager.MoveMouseTo(InputManager.CurrentState.Mouse.Position + new Vector2(-1, 0)), 10, true));
AddWaitStep("move", 10);
AddStep("stop move", () => moveFunction.Cancel());
AddAssert("at least 60 frames recorded", () => replay.Frames.Count > 60);
}
[Test]
public void TestLimitedFrameRate()
{
ScheduledDelegate moveFunction = null;
2021-04-19 08:02:59 +00:00
int initialFrameCount = 0;
2020-03-24 07:22:54 +00:00
AddStep("lower rate", () => recorder.RecordFrameRate = 2);
2021-04-19 08:02:59 +00:00
AddStep("count frames", () => initialFrameCount = replay.Frames.Count);
2020-03-24 07:22:54 +00:00
AddStep("move to center", () => InputManager.MoveMouseTo(recordingManager.ScreenSpaceDrawQuad.Centre));
AddStep("much move", () => moveFunction = Scheduler.AddDelayed(() =>
InputManager.MoveMouseTo(InputManager.CurrentState.Mouse.Position + new Vector2(-1, 0)), 10, true));
AddWaitStep("move", 10);
AddStep("stop move", () => moveFunction.Cancel());
2021-04-19 08:02:59 +00:00
AddAssert("less than 10 frames recorded", () => replay.Frames.Count - initialFrameCount < 10);
2020-03-24 07:22:54 +00:00
}
[Test]
public void TestLimitedFrameRateWithImportantFrames()
{
ScheduledDelegate moveFunction = null;
AddStep("lower rate", () => recorder.RecordFrameRate = 2);
AddStep("move to center", () => InputManager.MoveMouseTo(recordingManager.ScreenSpaceDrawQuad.Centre));
AddStep("much move with press", () => moveFunction = Scheduler.AddDelayed(() =>
{
InputManager.MoveMouseTo(InputManager.CurrentState.Mouse.Position + new Vector2(-1, 0));
2020-11-05 14:41:56 +00:00
InputManager.Click(MouseButton.Left);
2020-03-24 07:22:54 +00:00
}, 10, true));
AddWaitStep("move", 10);
AddStep("stop move", () => moveFunction.Cancel());
AddAssert("at least 60 frames recorded", () => replay.Frames.Count > 60);
2020-03-23 08:33:02 +00:00
}
2020-03-23 09:50:16 +00:00
protected override void Update()
2020-03-23 08:33:02 +00:00
{
2020-03-23 09:50:16 +00:00
base.Update();
2020-03-24 07:22:54 +00:00
playbackManager?.ReplayInputHandler.SetFrameFromTime(Time.Current - 100);
2020-03-23 09:50:16 +00:00
}
2020-03-23 08:33:02 +00:00
[TearDownSteps]
public void TearDown()
{
AddStep("stop recorder", () => recorder.Expire());
}
2020-03-24 06:54:04 +00:00
public class TestFramedReplayInputHandler : FramedReplayInputHandler<TestReplayFrame>
2020-03-23 09:50:16 +00:00
{
2020-03-24 06:54:04 +00:00
public TestFramedReplayInputHandler(Replay replay)
: base(replay)
{
}
2020-03-23 09:50:16 +00:00
2020-07-22 10:53:45 +00:00
public override void CollectPendingInputs(List<IInput> inputs)
2020-03-23 08:33:02 +00:00
{
inputs.Add(new MousePositionAbsoluteInput { Position = GamefieldToScreenSpace(CurrentFrame?.Position ?? Vector2.Zero) });
inputs.Add(new ReplayState<TestAction> { PressedActions = CurrentFrame?.Actions ?? new List<TestAction>() });
2020-03-24 06:54:04 +00:00
}
2020-03-23 09:50:16 +00:00
}
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
public class TestInputConsumer : CompositeDrawable, IKeyBindingHandler<TestAction>
2020-03-23 09:50:16 +00:00
{
2020-03-24 06:54:04 +00:00
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => Parent.ReceivePositionalInputAt(screenSpacePos);
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
private readonly Box box;
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
public TestInputConsumer()
2020-03-23 08:33:02 +00:00
{
2020-03-24 06:54:04 +00:00
Size = new Vector2(30);
Origin = Anchor.Centre;
InternalChildren = new Drawable[]
2020-03-23 09:50:16 +00:00
{
2020-03-24 06:54:04 +00:00
box = new Box
{
Colour = Color4.Black,
RelativeSizeAxes = Axes.Both,
},
};
}
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
protected override bool OnMouseMove(MouseMoveEvent e)
{
Position = e.MousePosition;
return base.OnMouseMove(e);
}
2020-03-23 09:50:16 +00:00
2021-09-16 09:26:12 +00:00
public bool OnPressed(KeyBindingPressEvent<TestAction> e)
2020-03-24 06:54:04 +00:00
{
box.Colour = Color4.White;
return true;
}
2020-03-23 09:50:16 +00:00
2021-09-16 09:26:12 +00:00
public void OnReleased(KeyBindingReleaseEvent<TestAction> e)
2020-03-24 06:54:04 +00:00
{
box.Colour = Color4.Black;
}
2020-03-23 09:50:16 +00:00
}
2020-03-24 06:54:04 +00:00
public class TestRulesetInputManager : RulesetInputManager<TestAction>
2020-03-23 09:50:16 +00:00
{
2020-03-24 06:54:04 +00:00
public TestRulesetInputManager(RulesetInfo ruleset, int variant, SimultaneousBindingMode unique)
: base(ruleset, variant, unique)
{
}
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
protected override KeyBindingContainer<TestAction> CreateKeyBindingContainer(RulesetInfo ruleset, int variant, SimultaneousBindingMode unique)
=> new TestKeyBindingContainer();
2020-03-23 08:33:02 +00:00
2020-03-24 06:54:04 +00:00
internal class TestKeyBindingContainer : KeyBindingContainer<TestAction>
2020-03-23 08:33:02 +00:00
{
public override IEnumerable<IKeyBinding> DefaultKeyBindings => new[]
2020-03-24 06:54:04 +00:00
{
new KeyBinding(InputKey.MouseLeft, TestAction.Down),
};
}
2020-03-23 09:50:16 +00:00
}
2020-03-24 06:54:04 +00:00
public class TestReplayFrame : ReplayFrame
{
public Vector2 Position;
2020-03-23 09:50:16 +00:00
2020-03-24 06:54:04 +00:00
public List<TestAction> Actions = new List<TestAction>();
2020-03-23 09:50:16 +00:00
2020-03-24 06:54:04 +00:00
public TestReplayFrame(double time, Vector2 position, params TestAction[] actions)
: base(time)
{
Position = position;
Actions.AddRange(actions);
}
2020-03-23 09:50:16 +00:00
}
2020-03-24 06:54:04 +00:00
public enum TestAction
2020-03-23 09:50:16 +00:00
{
2020-03-24 06:54:04 +00:00
Down,
2020-03-23 09:50:16 +00:00
}
2020-03-24 06:54:04 +00:00
internal class TestReplayRecorder : ReplayRecorder<TestAction>
{
public TestReplayRecorder(Score target)
2020-03-24 06:54:04 +00:00
: base(target)
{
}
protected override ReplayFrame HandleFrame(Vector2 mousePosition, List<TestAction> actions, ReplayFrame previousFrame)
2020-10-22 10:31:56 +00:00
=> new TestReplayFrame(Time.Current, mousePosition, actions.ToArray());
2020-03-24 06:54:04 +00:00
}
2020-03-23 09:50:16 +00:00
}
2020-03-23 08:33:02 +00:00
}