2019-01-24 16:43:03 +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.
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2022-06-17 15:37:17 +08:00
|
|
|
|
#nullable disable
|
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
using System;
|
2020-09-21 18:39:54 +08:00
|
|
|
|
using osu.Framework.Allocation;
|
2020-07-29 19:01:01 +08:00
|
|
|
|
using osu.Framework.Bindables;
|
2017-02-14 17:55:54 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
2018-10-02 11:02:47 +08:00
|
|
|
|
using osu.Framework.Input.Events;
|
2020-01-09 03:21:13 +08:00
|
|
|
|
using osu.Framework.Utils;
|
2020-11-15 06:23:11 +08:00
|
|
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
2020-12-04 19:21:53 +08:00
|
|
|
|
using osu.Game.Rulesets.Osu.Objects.Drawables;
|
2020-09-21 18:39:54 +08:00
|
|
|
|
using osu.Game.Screens.Play;
|
2020-07-29 19:01:01 +08:00
|
|
|
|
using osuTK;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-12-04 19:21:53 +08:00
|
|
|
|
namespace osu.Game.Rulesets.Osu.Skinning.Default
|
2017-02-14 17:55:54 +08:00
|
|
|
|
{
|
2020-07-29 19:01:01 +08:00
|
|
|
|
public class SpinnerRotationTracker : CircularContainer
|
2017-02-14 17:55:54 +08:00
|
|
|
|
{
|
2017-12-27 00:25:18 +08:00
|
|
|
|
public override bool IsPresent => true; // handle input when hidden
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-11-05 13:40:48 +08:00
|
|
|
|
private readonly DrawableSpinner drawableSpinner;
|
|
|
|
|
|
|
|
|
|
public SpinnerRotationTracker(DrawableSpinner drawableSpinner)
|
2017-02-14 17:55:54 +08:00
|
|
|
|
{
|
2020-11-05 13:40:48 +08:00
|
|
|
|
this.drawableSpinner = drawableSpinner;
|
2020-11-15 06:23:11 +08:00
|
|
|
|
drawableSpinner.HitObjectApplied += resetState;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2018-09-26 13:01:15 +08:00
|
|
|
|
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => true;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-07-29 19:01:01 +08:00
|
|
|
|
public bool Tracking { get; set; }
|
2019-02-28 12:31:40 +08:00
|
|
|
|
|
2020-07-30 18:34:59 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the spinning is spinning at a reasonable speed to be considered visually spinning.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public readonly BindableBool IsSpinning = new BindableBool();
|
|
|
|
|
|
2020-03-29 13:31:03 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether currently in the correct time range to allow spinning.
|
|
|
|
|
/// </summary>
|
2020-11-05 13:40:48 +08:00
|
|
|
|
private bool isSpinnableTime => drawableSpinner.HitObject.StartTime <= Time.Current && drawableSpinner.HitObject.EndTime > Time.Current;
|
2020-02-08 08:59:35 +08:00
|
|
|
|
|
2018-10-02 11:02:47 +08:00
|
|
|
|
protected override bool OnMouseMove(MouseMoveEvent e)
|
2017-02-14 17:55:54 +08:00
|
|
|
|
{
|
2018-09-19 19:52:57 +08:00
|
|
|
|
mousePosition = Parent.ToLocalSpace(e.ScreenSpaceMousePosition);
|
2018-10-02 11:02:47 +08:00
|
|
|
|
return base.OnMouseMove(e);
|
2017-02-14 17:55:54 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
private Vector2 mousePosition;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
private float lastAngle;
|
|
|
|
|
private float currentRotation;
|
2019-09-08 21:14:14 +08:00
|
|
|
|
|
2017-05-18 18:40:20 +08:00
|
|
|
|
private bool rotationTransferred;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-09-21 21:30:14 +08:00
|
|
|
|
[Resolved(canBeNull: true)]
|
2020-09-21 18:39:54 +08:00
|
|
|
|
private GameplayClock gameplayClock { get; set; }
|
|
|
|
|
|
2017-02-14 17:55:54 +08:00
|
|
|
|
protected override void Update()
|
|
|
|
|
{
|
|
|
|
|
base.Update();
|
2021-10-27 12:04:41 +08:00
|
|
|
|
float thisAngle = -MathUtils.RadiansToDegrees(MathF.Atan2(mousePosition.X - DrawSize.X / 2, mousePosition.Y - DrawSize.Y / 2));
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2021-10-27 12:04:41 +08:00
|
|
|
|
float delta = thisAngle - lastAngle;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-07-29 19:01:01 +08:00
|
|
|
|
if (Tracking)
|
|
|
|
|
AddRotation(delta);
|
2020-02-05 14:23:59 +08:00
|
|
|
|
|
2020-02-08 09:51:32 +08:00
|
|
|
|
lastAngle = thisAngle;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2020-07-30 18:34:59 +08:00
|
|
|
|
IsSpinning.Value = isSpinnableTime && Math.Abs(currentRotation / 2 - Rotation) > 5f;
|
|
|
|
|
|
|
|
|
|
Rotation = (float)Interpolation.Damp(Rotation, currentRotation / 2, 0.99, Math.Abs(Time.Elapsed));
|
2017-02-14 17:55:54 +08:00
|
|
|
|
}
|
2020-02-05 14:23:59 +08:00
|
|
|
|
|
2020-03-29 13:31:03 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Rotate the disc by the provided angle (in addition to any existing rotation).
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// Will be a no-op if not a valid time to spin.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
/// <param name="angle">The delta angle.</param>
|
2020-07-29 19:01:01 +08:00
|
|
|
|
public void AddRotation(float angle)
|
2020-02-05 14:23:59 +08:00
|
|
|
|
{
|
2020-03-29 13:31:03 +08:00
|
|
|
|
if (!isSpinnableTime)
|
|
|
|
|
return;
|
|
|
|
|
|
2020-02-05 14:23:59 +08:00
|
|
|
|
if (!rotationTransferred)
|
|
|
|
|
{
|
|
|
|
|
currentRotation = Rotation * 2;
|
|
|
|
|
rotationTransferred = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (angle > 180)
|
|
|
|
|
{
|
|
|
|
|
lastAngle += 360;
|
|
|
|
|
angle -= 360;
|
|
|
|
|
}
|
|
|
|
|
else if (-angle > 180)
|
|
|
|
|
{
|
|
|
|
|
lastAngle -= 360;
|
|
|
|
|
angle += 360;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
currentRotation += angle;
|
2020-08-11 04:17:47 +08:00
|
|
|
|
// rate has to be applied each frame, because it's not guaranteed to be constant throughout playback
|
|
|
|
|
// (see: ModTimeRamp)
|
2020-11-15 04:10:12 +08:00
|
|
|
|
drawableSpinner.Result.RateAdjustedRotation += (float)(Math.Abs(angle) * (gameplayClock?.TrueGameplayRate ?? Clock.Rate));
|
2020-02-05 14:23:59 +08:00
|
|
|
|
}
|
2020-11-15 06:23:11 +08:00
|
|
|
|
|
|
|
|
|
private void resetState(DrawableHitObject obj)
|
|
|
|
|
{
|
|
|
|
|
Tracking = false;
|
|
|
|
|
IsSpinning.Value = false;
|
|
|
|
|
mousePosition = default;
|
|
|
|
|
lastAngle = currentRotation = Rotation = 0;
|
|
|
|
|
rotationTransferred = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
|
{
|
|
|
|
|
base.Dispose(isDisposing);
|
|
|
|
|
|
|
|
|
|
if (drawableSpinner != null)
|
|
|
|
|
drawableSpinner.HitObjectApplied -= resetState;
|
|
|
|
|
}
|
2017-02-14 17:55:54 +08:00
|
|
|
|
}
|
|
|
|
|
}
|