1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-19 00:43:20 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSliderRepeat.cs

167 lines
5.8 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 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
2018-04-13 17:19:50 +08:00
using System;
using System.Collections.Generic;
using JetBrains.Annotations;
using osu.Framework.Allocation;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2020-01-09 12:43:44 +08:00
using osu.Framework.Utils;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Objects.Drawables;
2020-12-04 19:21:53 +08:00
using osu.Game.Rulesets.Osu.Skinning.Default;
using osu.Game.Skinning;
2019-09-06 14:24:00 +08:00
using osuTK;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
public partial class DrawableSliderRepeat : DrawableOsuHitObject, ITrackSnaking
2018-04-13 17:19:50 +08:00
{
2020-11-12 14:59:48 +08:00
public new SliderRepeat HitObject => (SliderRepeat)base.HitObject;
2018-04-13 17:19:50 +08:00
[CanBeNull]
public Slider Slider => DrawableSlider?.HitObject;
public DrawableSlider DrawableSlider => (DrawableSlider)ParentHitObject;
2018-04-13 17:19:50 +08:00
private double animDuration;
public SkinnableDrawable CirclePiece { get; private set; }
public SkinnableDrawable Arrow { get; private set; }
2020-11-05 12:51:46 +08:00
private Drawable scaleContainer;
2020-11-12 14:59:48 +08:00
public DrawableSliderRepeat()
: base(null)
{
}
public DrawableSliderRepeat(SliderRepeat sliderRepeat)
2020-03-19 13:42:02 +08:00
: base(sliderRepeat)
2018-04-13 17:19:50 +08:00
{
2020-11-05 12:51:46 +08:00
}
2018-04-13 17:19:50 +08:00
2020-11-05 12:51:46 +08:00
[BackgroundDependencyLoader]
private void load()
{
2018-04-13 17:19:50 +08:00
Origin = Anchor.Centre;
2023-09-20 11:48:15 +08:00
Size = OsuHitObject.OBJECT_DIMENSIONS;
2018-04-13 17:19:50 +08:00
AddInternal(scaleContainer = new Container
{
RelativeSizeAxes = Axes.Both,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Children = new Drawable[]
{
// no default for this; only visible in legacy skins.
CirclePiece = new SkinnableDrawable(new OsuSkinComponentLookup(OsuSkinComponents.SliderTailHitCircle), _ => Empty())
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
Arrow = new SkinnableDrawable(new OsuSkinComponentLookup(OsuSkinComponents.ReverseArrow), _ => new DefaultReverseArrow())
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
}
});
2018-04-13 17:19:50 +08:00
2020-11-12 14:59:48 +08:00
ScaleBindable.BindValueChanged(scale => scaleContainer.Scale = new Vector2(scale.NewValue));
}
2020-12-03 18:46:42 +08:00
protected override void OnApply()
2020-11-12 14:59:48 +08:00
{
2020-12-03 18:46:42 +08:00
base.OnApply();
2020-11-12 14:59:48 +08:00
2020-12-03 19:03:39 +08:00
Position = HitObject.Position - DrawableSlider.Position;
}
protected override void CheckForResult(bool userTriggered, double timeOffset) => DrawableSlider.SliderInputManager.TryJudgeNestedObject(this, timeOffset);
2018-04-13 17:19:50 +08:00
2019-07-22 14:33:12 +08:00
protected override void UpdateInitialTransforms()
2018-04-13 17:19:50 +08:00
{
// When snaking in is enabled, the first end circle needs to be delayed until the snaking completes.
bool delayFadeIn = DrawableSlider.SliderBody?.SnakingIn.Value == true && HitObject.RepeatIndex == 0;
2020-11-12 14:59:48 +08:00
animDuration = Math.Min(300, HitObject.SpanDuration);
2018-04-13 17:19:50 +08:00
this
.FadeOut()
.Delay(delayFadeIn ? (Slider?.TimePreempt ?? 0) / 3 : 0)
.FadeIn(HitObject.RepeatIndex == 0 ? HitObject.TimeFadeIn : animDuration);
2018-04-13 17:19:50 +08:00
}
2020-11-04 15:19:07 +08:00
protected override void UpdateHitStateTransforms(ArmedState state)
2018-04-13 17:19:50 +08:00
{
2020-11-04 15:19:07 +08:00
base.UpdateHitStateTransforms(state);
2019-09-13 17:49:21 +08:00
2018-04-13 17:19:50 +08:00
switch (state)
{
case ArmedState.Idle:
this.Delay(HitObject.TimePreempt).FadeOut();
break;
2019-04-01 11:44:46 +08:00
2018-04-13 17:19:50 +08:00
case ArmedState.Miss:
this.FadeOut(animDuration);
break;
2019-04-01 11:44:46 +08:00
2018-04-13 17:19:50 +08:00
case ArmedState.Hit:
this.FadeOut(animDuration, Easing.Out);
2018-04-13 17:19:50 +08:00
break;
}
}
private bool hasRotation;
2018-04-13 17:19:50 +08:00
public void UpdateSnakingPosition(Vector2 start, Vector2 end)
{
2020-03-30 17:35:01 +08:00
// When the repeat is hit, the arrow should fade out on spot rather than following the slider
2020-03-29 05:12:13 +08:00
if (IsHit) return;
2020-11-12 14:59:48 +08:00
bool isRepeatAtEnd = HitObject.RepeatIndex % 2 == 0;
2020-12-03 19:03:39 +08:00
List<Vector2> curve = ((PlaySliderBody)DrawableSlider.Body.Drawable).CurrentCurve;
2018-04-13 17:19:50 +08: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 17:19:50 +08: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 17:19:50 +08:00
continue;
aimRotationVector = curve[i];
2018-04-13 17:19:50 +08:00
break;
}
float aimRotation = MathUtils.RadiansToDegrees(MathF.Atan2(aimRotationVector.Y - Position.Y, aimRotationVector.X - Position.X));
while (Math.Abs(aimRotation - Arrow.Rotation) > 180)
aimRotation += aimRotation < Arrow.Rotation ? 360 : -360;
// The clock may be paused in a scenario like the editor.
if (!hasRotation || !Clock.IsRunning)
{
Arrow.Rotation = aimRotation;
hasRotation = true;
}
else
{
2018-07-31 15:47:13 +08:00
// If we're already snaking, interpolate to smooth out sharp curves (linear sliders, mainly).
Arrow.Rotation = Interpolation.ValueAt(Math.Clamp(Clock.ElapsedFrameTime, 0, 100), Arrow.Rotation, aimRotation, 0, 50, Easing.OutQuint);
}
2018-04-13 17:19:50 +08:00
}
}
}