1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-24 04:07:26 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Objects/Drawables/DrawableSpinner.cs

299 lines
10 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
using System;
2018-04-13 17:19:50 +08:00
using System.Linq;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Osu.Objects.Drawables.Pieces;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
using osu.Framework.Graphics.Primitives;
using osu.Game.Rulesets.Objects;
using osu.Framework.Utils;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Scoring;
2019-09-06 14:24:00 +08:00
using osu.Game.Screens.Ranking;
using osu.Game.Skinning;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
public class DrawableSpinner : DrawableOsuHitObject
{
protected readonly Spinner Spinner;
private readonly Container<DrawableSpinnerTick> ticks;
2018-04-13 17:19:50 +08:00
public readonly SpinnerDisc Disc;
public readonly SpinnerTicks Ticks;
2019-12-18 08:04:37 +08:00
public readonly SpinnerSpmCounter SpmCounter;
private readonly SpinnerBonusDisplay bonusDisplay;
2018-04-13 17:19:50 +08:00
private readonly Container mainContainer;
public readonly SkinnableDrawable Background;
private readonly SkinnableDrawable circleContainer;
2018-04-13 17:19:50 +08:00
private readonly Color4 baseColour = Color4Extensions.FromHex(@"002c3c");
private readonly Color4 fillColour = Color4Extensions.FromHex(@"005b7c");
2018-04-13 17:19:50 +08:00
2018-11-09 12:58:46 +08:00
private readonly IBindable<Vector2> positionBindable = new Bindable<Vector2>();
2018-04-13 17:19:50 +08:00
private Color4 normalColour;
private Color4 completeColour;
2019-02-28 12:31:40 +08:00
public DrawableSpinner(Spinner s)
: base(s)
2018-04-13 17:19:50 +08:00
{
Origin = Anchor.Centre;
Position = s.Position;
RelativeSizeAxes = Axes.Both;
// we are slightly bigger than our parent, to clip the top and bottom of the circle
Height = 1.3f;
Spinner = s;
InternalChildren = new Drawable[]
{
ticks = new Container<DrawableSpinnerTick>(),
circleContainer = new SkinnableDrawable(new OsuSkinComponent(OsuSkinComponents.SpinnerCentre), _ => new DefaultSpinnerCentre())
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
2018-04-13 17:19:50 +08:00
},
mainContainer = new AspectContainer
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
RelativeSizeAxes = Axes.Y,
2018-04-20 17:19:17 +08:00
Children = new[]
2018-04-13 17:19:50 +08:00
{
Background = new SkinnableDrawable(new OsuSkinComponent(OsuSkinComponents.SpinnerBackground), _ => new DefaultSpinnerBackground()),
2018-04-13 17:19:50 +08:00
Disc = new SpinnerDisc(Spinner)
{
Scale = Vector2.Zero,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
circleContainer.CreateProxy(),
Ticks = new SpinnerTicks
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
},
}
},
2019-12-18 08:04:37 +08:00
SpmCounter = new SpinnerSpmCounter
2018-04-13 17:19:50 +08:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Y = 120,
Alpha = 0
},
bonusDisplay = new SpinnerBonusDisplay
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Y = -120,
2018-04-13 17:19:50 +08:00
}
};
}
protected override void AddNestedHitObject(DrawableHitObject hitObject)
{
base.AddNestedHitObject(hitObject);
switch (hitObject)
{
case DrawableSpinnerTick tick:
ticks.Add(tick);
break;
}
}
protected override void ClearNestedHitObjects()
{
base.ClearNestedHitObjects();
ticks.Clear();
}
protected override DrawableHitObject CreateNestedHitObject(HitObject hitObject)
{
switch (hitObject)
{
case SpinnerBonusTick bonusTick:
return new DrawableSpinnerBonusTick(bonusTick);
case SpinnerTick tick:
return new DrawableSpinnerTick(tick);
}
return base.CreateNestedHitObject(hitObject);
2018-11-09 12:58:46 +08:00
}
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
normalColour = baseColour;
2020-07-20 16:48:35 +08:00
completeColour = colours.YellowLight;
2018-11-09 12:58:46 +08:00
2020-07-20 16:48:35 +08:00
Ticks.AccentColour = normalColour;
2018-11-09 12:58:46 +08:00
Disc.AccentColour = fillColour;
positionBindable.BindValueChanged(pos => Position = pos.NewValue);
2018-11-09 12:58:46 +08:00
positionBindable.BindTo(HitObject.PositionBindable);
2018-04-13 17:19:50 +08:00
}
public float Progress => Math.Clamp(Disc.CumulativeRotation / 360 / Spinner.SpinsRequired, 0, 1);
2018-04-13 17:19:50 +08:00
protected override void CheckForResult(bool userTriggered, double timeOffset)
2018-04-13 17:19:50 +08:00
{
if (Time.Current < HitObject.StartTime) return;
if (Progress >= 1 && !Disc.Complete)
{
Disc.Complete = true;
transformFillColour(completeColour, 200);
2018-04-13 17:19:50 +08:00
}
if (userTriggered || Time.Current < Spinner.EndTime)
return;
// Trigger a miss result for remaining ticks to avoid infinite gameplay.
foreach (var tick in ticks.Where(t => !t.IsHit))
2020-07-21 18:48:44 +08:00
tick.TriggerResult(false);
ApplyResult(r =>
2018-04-13 17:19:50 +08:00
{
if (Progress >= 1)
r.Type = HitResult.Great;
2018-04-13 17:19:50 +08:00
else if (Progress > .9)
r.Type = HitResult.Good;
2018-04-13 17:19:50 +08:00
else if (Progress > .75)
r.Type = HitResult.Meh;
2018-04-13 17:19:50 +08:00
else if (Time.Current >= Spinner.EndTime)
r.Type = HitResult.Miss;
});
2018-04-13 17:19:50 +08:00
}
protected override void Update()
{
base.Update();
if (HandleUserInput)
Disc.Tracking = OsuActionInputManager?.PressedActions.Any(x => x == OsuAction.LeftButton || x == OsuAction.RightButton) ?? false;
2018-04-13 17:19:50 +08:00
}
private float relativeHeight => ToScreenSpace(new RectangleF(0, 0, OsuHitObject.OBJECT_RADIUS, OsuHitObject.OBJECT_RADIUS)).Height / mainContainer.DrawHeight;
2018-04-13 17:19:50 +08:00
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
if (!SpmCounter.IsPresent && Disc.Tracking)
SpmCounter.FadeIn(HitObject.TimeFadeIn);
2018-04-13 17:19:50 +08:00
Ticks.Rotation = Disc.Rotation;
SpmCounter.SetRotation(Disc.CumulativeRotation);
updateBonusScore();
float relativeCircleScale = Spinner.Scale * relativeHeight;
float targetScale = relativeCircleScale + (1 - relativeCircleScale) * Progress;
Disc.Scale = new Vector2((float)Interpolation.Lerp(Disc.Scale.X, targetScale, Math.Clamp(Math.Abs(Time.Elapsed) / 100, 0, 1)));
2018-04-13 17:19:50 +08:00
}
private int wholeSpins;
private void updateBonusScore()
{
if (ticks.Count == 0)
return;
int spins = (int)(Disc.CumulativeRotation / 360);
2020-07-21 18:21:30 +08:00
if (spins < wholeSpins)
{
2020-07-21 18:21:30 +08:00
// rewinding, silently handle
wholeSpins = spins;
return;
}
2020-07-21 18:21:30 +08:00
while (wholeSpins != spins)
{
var tick = ticks.FirstOrDefault(t => !t.IsHit);
2020-07-21 18:21:30 +08:00
// tick may be null if we've hit the spin limit.
if (tick != null)
{
2020-07-21 18:48:44 +08:00
tick.TriggerResult(true);
2020-07-21 18:21:30 +08:00
if (tick is DrawableSpinnerBonusTick)
bonusDisplay.SetBonusCount(spins - Spinner.SpinsRequired);
}
2020-07-21 18:21:30 +08:00
wholeSpins++;
}
}
2019-07-22 14:33:12 +08:00
protected override void UpdateInitialTransforms()
2018-04-13 17:19:50 +08:00
{
2019-07-22 14:33:12 +08:00
base.UpdateInitialTransforms();
2018-04-13 17:19:50 +08:00
2020-07-20 14:19:17 +08:00
circleContainer.ScaleTo(0);
mainContainer.ScaleTo(0);
2018-04-13 17:19:50 +08:00
2020-07-20 14:19:17 +08:00
using (BeginDelayedSequence(HitObject.TimePreempt / 2, true))
{
float phaseOneScale = Spinner.Scale * 0.7f;
2020-07-20 14:19:17 +08:00
circleContainer.ScaleTo(phaseOneScale, HitObject.TimePreempt / 4, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
2020-07-20 14:19:17 +08:00
mainContainer
.ScaleTo(phaseOneScale * relativeHeight * 1.6f, HitObject.TimePreempt / 4, Easing.OutQuint)
.RotateTo((float)(25 * Spinner.Duration / 2000), HitObject.TimePreempt + Spinner.Duration);
2018-04-13 17:19:50 +08:00
2020-07-20 14:19:17 +08:00
using (BeginDelayedSequence(HitObject.TimePreempt / 2, true))
{
circleContainer.ScaleTo(Spinner.Scale, 400, Easing.OutQuint);
mainContainer.ScaleTo(1, 400, Easing.OutQuint);
2020-07-20 14:19:17 +08:00
}
}
2018-04-13 17:19:50 +08:00
}
2019-07-22 14:33:12 +08:00
protected override void UpdateStateTransforms(ArmedState state)
2018-04-13 17:19:50 +08:00
{
2019-09-13 17:49:21 +08:00
base.UpdateStateTransforms(state);
using (BeginDelayedSequence(Spinner.Duration, true))
2018-04-13 17:19:50 +08:00
{
this.FadeOut(160);
2019-04-01 11:44:46 +08:00
switch (state)
{
case ArmedState.Hit:
transformFillColour(completeColour, 0);
this.ScaleTo(Scale * 1.2f, 320, Easing.Out);
mainContainer.RotateTo(mainContainer.Rotation + 180, 320);
break;
case ArmedState.Miss:
this.ScaleTo(Scale * 0.8f, 320, Easing.In);
break;
}
2018-04-13 17:19:50 +08:00
}
}
private void transformFillColour(Colour4 colour, double duration)
{
Disc.FadeAccent(colour, duration);
2020-07-20 16:48:35 +08:00
Ticks.FadeAccent(colour, duration);
}
2018-04-13 17:19:50 +08:00
}
}