osu/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSliderRepeat.cs

132 lines
4.5 KiB
C#
Raw Normal View History

// 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;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics;
2020-01-09 04:43:44 +00:00
using osu.Framework.Utils;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Osu.Objects.Drawables.Pieces;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.Scoring;
2019-09-06 06:24:00 +00:00
using osuTK;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
public class DrawableSliderRepeat : DrawableOsuHitObject, ITrackSnaking
2018-04-13 09:19:50 +00:00
{
2020-03-19 05:42:02 +00:00
private readonly SliderRepeat sliderRepeat;
2018-04-13 09:19:50 +00:00
private readonly DrawableSlider drawableSlider;
private double animDuration;
private readonly Drawable scaleContainer;
2020-03-19 05:42:02 +00:00
public DrawableSliderRepeat(SliderRepeat sliderRepeat, DrawableSlider drawableSlider)
: base(sliderRepeat)
2018-04-13 09:19:50 +00:00
{
2020-03-19 05:42:02 +00:00
this.sliderRepeat = sliderRepeat;
2018-04-13 09:19:50 +00:00
this.drawableSlider = drawableSlider;
Size = new Vector2(OsuHitObject.OBJECT_RADIUS * 2);
2018-04-13 09:19:50 +00:00
Origin = Anchor.Centre;
2020-01-14 09:14:31 +00:00
InternalChild = scaleContainer = new ReverseArrowPiece();
2018-04-13 09:19:50 +00:00
}
private readonly IBindable<float> scaleBindable = new BindableFloat();
[BackgroundDependencyLoader]
private void load()
{
scaleBindable.BindValueChanged(scale => scaleContainer.Scale = new Vector2(scale.NewValue), true);
scaleBindable.BindTo(HitObject.ScaleBindable);
}
protected override void CheckForResult(bool userTriggered, double timeOffset)
2018-04-13 09:19:50 +00:00
{
2020-03-19 05:42:02 +00:00
if (sliderRepeat.StartTime <= Time.Current)
ApplyResult(r => r.Type = drawableSlider.Tracking.Value ? HitResult.Great : HitResult.Miss);
2018-04-13 09:19:50 +00:00
}
2019-07-22 06:33:12 +00:00
protected override void UpdateInitialTransforms()
2018-04-13 09:19:50 +00:00
{
2020-03-19 05:42:02 +00:00
animDuration = Math.Min(300, sliderRepeat.SpanDuration);
2018-04-13 09:19:50 +00:00
this.Animate(
d => d.FadeIn(animDuration),
d => d.ScaleTo(0.5f).ScaleTo(1f, animDuration * 2, Easing.OutElasticHalf)
);
2018-04-13 09:19:50 +00:00
}
2019-07-22 06:33:12 +00:00
protected override void UpdateStateTransforms(ArmedState state)
2018-04-13 09:19:50 +00:00
{
2019-09-13 09:49:21 +00:00
base.UpdateStateTransforms(state);
2018-04-13 09:19:50 +00:00
switch (state)
{
case ArmedState.Idle:
this.Delay(HitObject.TimePreempt).FadeOut();
break;
2019-04-01 03:44:46 +00:00
2018-04-13 09:19:50 +00:00
case ArmedState.Miss:
this.FadeOut(animDuration);
break;
2019-04-01 03:44:46 +00:00
2018-04-13 09:19:50 +00:00
case ArmedState.Hit:
2020-01-04 07:21:48 +00:00
this.FadeOut(animDuration, Easing.Out)
2018-04-13 09:19:50 +00:00
.ScaleTo(Scale * 1.5f, animDuration, Easing.Out);
break;
}
}
private bool hasRotation;
2018-04-13 09:19:50 +00:00
public void UpdateSnakingPosition(Vector2 start, Vector2 end)
{
2020-03-30 09:35:01 +00:00
// When the repeat is hit, the arrow should fade out on spot rather than following the slider
2020-03-28 21:12:13 +00:00
if (IsHit) return;
2020-03-19 05:42:02 +00:00
bool isRepeatAtEnd = sliderRepeat.RepeatIndex % 2 == 0;
List<Vector2> curve = ((PlaySliderBody)drawableSlider.Body.Drawable).CurrentCurve;
2018-04-13 09:19:50 +00:00
Position = isRepeatAtEnd ? end : start;
if (curve.Count < 2)
return;
int searchStart = isRepeatAtEnd ? curve.Count - 1 : 0;
int direction = isRepeatAtEnd ? -1 : 1;
Vector2 aimRotationVector = Vector2.Zero;
2018-04-13 09:19:50 +00:00
// find the next vector2 in the curve which is not equal to our current position to infer a rotation.
for (int i = searchStart; i >= 0 && i < curve.Count; i += direction)
{
if (Precision.AlmostEquals(curve[i], Position))
2018-04-13 09:19:50 +00:00
continue;
aimRotationVector = curve[i];
2018-04-13 09:19:50 +00:00
break;
}
float aimRotation = MathUtils.RadiansToDegrees(MathF.Atan2(aimRotationVector.Y - Position.Y, aimRotationVector.X - Position.X));
while (Math.Abs(aimRotation - Rotation) > 180)
aimRotation += aimRotation < Rotation ? 360 : -360;
if (!hasRotation)
{
Rotation = aimRotation;
hasRotation = true;
}
else
{
2018-07-31 07:47:13 +00:00
// If we're already snaking, interpolate to smooth out sharp curves (linear sliders, mainly).
Rotation = Interpolation.ValueAt(Math.Clamp(Clock.ElapsedFrameTime, 0, 100), Rotation, aimRotation, 0, 50, Easing.OutQuint);
}
2018-04-13 09:19:50 +00:00
}
}
}