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

103 lines
3.5 KiB
C#
Raw Normal View History

2018-01-05 11:21:19 +00:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
2017-09-26 16:13:34 +00:00
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System;
using osu.Framework.Graphics;
using osu.Game.Rulesets.Objects.Drawables;
using OpenTK;
using osu.Game.Graphics;
using osu.Game.Rulesets.Osu.Judgements;
using osu.Game.Rulesets.Scoring;
2017-09-26 16:13:34 +00:00
2017-09-26 16:21:39 +00:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
2017-09-26 16:13:34 +00:00
{
public class DrawableRepeatPoint : DrawableOsuHitObject, ITrackSnaking
2017-09-26 16:13:34 +00:00
{
2017-09-27 15:28:44 +00:00
private readonly RepeatPoint repeatPoint;
2017-09-26 16:13:34 +00:00
private readonly DrawableSlider drawableSlider;
2018-01-24 21:41:51 +00:00
/// <summary>
/// Whether this repeat point is at the end of the slider's curve.
2018-01-24 21:41:51 +00:00
/// </summary>
private bool isRepeatAtEnd => repeatPoint.RepeatIndex % 2 == 0;
2017-09-26 16:13:34 +00:00
private double animDuration;
2017-09-26 16:13:34 +00:00
public DrawableRepeatPoint(RepeatPoint repeatPoint, DrawableSlider drawableSlider)
: base(repeatPoint)
2017-09-26 16:13:34 +00:00
{
2017-09-27 15:28:44 +00:00
this.repeatPoint = repeatPoint;
2017-09-26 16:13:34 +00:00
this.drawableSlider = drawableSlider;
2018-01-23 10:31:37 +00:00
Size = new Vector2(45 * repeatPoint.Scale);
2017-09-26 16:13:34 +00:00
Blending = BlendingMode.Additive;
Origin = Anchor.Centre;
2017-09-26 16:13:34 +00:00
Children = new Drawable[]
{
new SpriteIcon
{
RelativeSizeAxes = Axes.Both,
2018-01-23 10:31:37 +00:00
Icon = FontAwesome.fa_chevron_right
2017-09-26 16:13:34 +00:00
}
};
}
protected override void CheckForJudgements(bool userTriggered, double timeOffset)
{
2017-09-27 15:28:44 +00:00
if (repeatPoint.StartTime <= Time.Current)
2017-09-26 16:13:34 +00:00
AddJudgement(new OsuJudgement { Result = drawableSlider.Tracking ? HitResult.Great : HitResult.Miss });
}
protected override void UpdatePreemptState()
{
animDuration = Math.Min(150, repeatPoint.SpanDuration / 2);
2017-09-26 16:13:34 +00:00
this.FadeIn(animDuration).ScaleTo(1.2f, animDuration / 2)
.Then()
.ScaleTo(1, animDuration / 2, Easing.Out);
2017-09-26 16:13:34 +00:00
}
protected override void UpdateCurrentState(ArmedState state)
{
switch (state)
{
case ArmedState.Idle:
this.Delay(HitObject.TimePreempt).FadeOut();
2017-09-26 16:13:34 +00:00
break;
case ArmedState.Miss:
this.FadeOut(animDuration);
2017-09-26 16:13:34 +00:00
break;
case ArmedState.Hit:
this.FadeOut(animDuration, Easing.OutQuint)
2018-02-06 08:46:56 +00:00
.ScaleTo(Scale * 1.5f, animDuration, Easing.Out);
2017-09-26 16:13:34 +00:00
break;
}
}
2018-01-23 10:31:37 +00:00
public void UpdateSnakingPosition(Vector2 start, Vector2 end)
{
2018-01-24 21:41:51 +00:00
Position = isRepeatAtEnd ? end : start;
2018-02-06 08:46:45 +00:00
var curve = drawableSlider.Body.CurrentCurve;
2018-02-06 08:46:45 +00:00
if (curve.Count < 2)
2018-01-23 10:31:37 +00:00
return;
2018-02-06 08:46:45 +00:00
int searchStart = isRepeatAtEnd ? curve.Count - 1 : 0;
int direction = isRepeatAtEnd ? -1 : 1;
// 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)
{
2018-02-06 08:46:45 +00:00
if (curve[i] == Position)
continue;
Rotation = MathHelper.RadiansToDegrees((float)Math.Atan2(curve[i].Y - Position.Y, curve[i].X - Position.X));
break;
}
2018-01-23 10:31:37 +00:00
}
2017-09-26 16:13:34 +00:00
}
}