1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-12 01:07:25 +08:00
osu-lazer/osu.Game/Rulesets/UI/ReplayRecorder.cs

115 lines
3.1 KiB
C#
Raw Normal View History

2020-03-23 18:03:42 +08: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.
using System;
2020-03-23 18:03:42 +08:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Allocation;
2020-03-23 18:03:42 +08:00
using osu.Framework.Graphics;
using osu.Framework.Input;
using osu.Framework.Input.Bindings;
using osu.Framework.Input.Events;
using osu.Game.Online.Spectator;
2020-03-23 18:03:42 +08:00
using osu.Game.Rulesets.Replays;
using osu.Game.Scoring;
using osu.Game.Screens.Play;
using osuTK;
2020-03-23 18:03:42 +08:00
namespace osu.Game.Rulesets.UI
{
public abstract class ReplayRecorder<T> : ReplayRecorder, IKeyBindingHandler<T>
where T : struct
{
private readonly Score target;
2020-03-23 18:03:42 +08:00
private readonly List<T> pressedActions = new List<T>();
private InputManager inputManager;
public int RecordFrameRate = 60;
2020-10-22 14:26:57 +08:00
[Resolved(canBeNull: true)]
private SpectatorClient spectatorClient { get; set; }
2020-10-22 14:26:57 +08:00
[Resolved]
2021-10-02 01:22:23 +08:00
private GameplayState gameplayState { get; set; }
2020-10-22 14:26:57 +08:00
protected ReplayRecorder(Score target)
2020-03-23 18:03:42 +08:00
{
this.target = target;
RelativeSizeAxes = Axes.Both;
Depth = float.MinValue;
}
protected override void LoadComplete()
{
base.LoadComplete();
inputManager = GetContainingInputManager();
2020-10-22 14:26:57 +08:00
2021-10-02 01:22:23 +08:00
spectatorClient?.BeginPlaying(gameplayState, target);
2020-10-22 14:26:57 +08:00
}
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
2022-02-01 14:51:41 +08:00
if (spectatorClient != null && gameplayState != null)
spectatorClient.EndPlaying(gameplayState);
2020-03-23 18:03:42 +08:00
}
2021-04-16 18:40:56 +08:00
protected override void Update()
{
base.Update();
recordFrame(false);
}
2020-03-23 18:03:42 +08:00
protected override bool OnMouseMove(MouseMoveEvent e)
{
recordFrame(false);
return base.OnMouseMove(e);
}
2021-09-16 17:26:12 +08:00
public bool OnPressed(KeyBindingPressEvent<T> e)
2020-03-23 18:03:42 +08:00
{
2021-09-16 17:26:12 +08:00
pressedActions.Add(e.Action);
2020-03-23 18:03:42 +08:00
recordFrame(true);
return false;
}
2021-09-16 17:26:12 +08:00
public void OnReleased(KeyBindingReleaseEvent<T> e)
2020-03-23 18:03:42 +08:00
{
2021-09-16 17:26:12 +08:00
pressedActions.Remove(e.Action);
2020-03-23 18:03:42 +08:00
recordFrame(true);
}
private void recordFrame(bool important)
{
var last = target.Replay.Frames.LastOrDefault();
2020-03-23 18:03:42 +08:00
if (!important && last != null && Time.Current - last.Time < (1000d / RecordFrameRate))
return;
var position = ScreenSpaceToGamefield?.Invoke(inputManager.CurrentState.Mouse.Position) ?? inputManager.CurrentState.Mouse.Position;
var frame = HandleFrame(position, pressedActions, last);
2020-03-23 18:03:42 +08:00
if (frame != null)
{
target.Replay.Frames.Add(frame);
spectatorClient?.HandleFrame(frame);
}
2020-03-23 18:03:42 +08:00
}
protected abstract ReplayFrame HandleFrame(Vector2 mousePosition, List<T> actions, ReplayFrame previousFrame);
2020-03-23 18:03:42 +08:00
}
public abstract class ReplayRecorder : Component
{
public Func<Vector2, Vector2> ScreenSpaceToGamefield;
2020-03-23 18:03:42 +08:00
}
}