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

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

335 lines
12 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
2022-06-17 07:37:17 +00:00
#nullable disable
using System;
2020-07-22 07:37:38 +00:00
using System.Linq;
using JetBrains.Annotations;
using osu.Framework.Allocation;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2020-11-19 11:40:30 +00:00
using osu.Game.Audio;
using osu.Game.Graphics.Containers;
2018-11-14 05:29:22 +00:00
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.Objects.Drawables;
2020-12-04 11:21:53 +00:00
using osu.Game.Rulesets.Osu.Skinning.Default;
using osu.Game.Rulesets.Scoring;
2018-12-07 21:24:24 +00:00
using osu.Game.Skinning;
using osuTK;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Osu.Objects.Drawables
{
public partial class DrawableSlider : DrawableOsuHitObject
{
2020-11-05 04:51:46 +00:00
public new Slider HitObject => (Slider)base.HitObject;
public DrawableSliderHead HeadCircle => headContainer.Child;
public DrawableSliderTail TailCircle => tailContainer.Child;
2018-04-13 09:19:50 +00:00
[Cached]
public DrawableSliderBall Ball { get; private set; }
2020-11-05 04:51:46 +00:00
public SkinnableDrawable Body { get; private set; }
2018-04-13 09:19:50 +00:00
private ShakeContainer shakeContainer;
/// <summary>
/// A target container which can be used to add top level elements to the slider's display.
/// Intended to be used for proxy purposes only.
/// </summary>
public Container OverlayElementContainer { get; private set; }
public override bool DisplayResult => !HitObject.OnlyJudgeNestedObjects;
[CanBeNull]
public PlaySliderBody SliderBody => Body.Drawable as PlaySliderBody;
2019-12-17 09:16:25 +00:00
public IBindable<int> PathVersion => pathVersion;
private readonly Bindable<int> pathVersion = new Bindable<int>();
2020-11-05 04:51:46 +00:00
private Container<DrawableSliderHead> headContainer;
private Container<DrawableSliderTail> tailContainer;
private Container<DrawableSliderTick> tickContainer;
private Container<DrawableSliderRepeat> repeatContainer;
2020-11-19 11:40:30 +00:00
private PausableSkinnableSound slidingSample;
2020-11-10 15:22:06 +00:00
public DrawableSlider()
: this(null)
{
}
public DrawableSlider([CanBeNull] Slider s = null)
: base(s)
{
Ball = new DrawableSliderBall
{
GetInitialHitAction = () => HeadCircle.HitAction,
BypassAutoSizeAxes = Axes.Both,
AlwaysPresent = true,
Alpha = 0
};
2020-11-05 04:51:46 +00:00
}
2018-04-13 09:19:50 +00:00
2020-11-05 04:51:46 +00:00
[BackgroundDependencyLoader]
private void load()
{
AddRangeInternal(new Drawable[]
{
shakeContainer = new ShakeContainer
{
ShakeDuration = 30,
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
Body = new SkinnableDrawable(new OsuSkinComponentLookup(OsuSkinComponents.SliderBody), _ => new DefaultSliderBody(), confineMode: ConfineMode.NoScaling),
tailContainer = new Container<DrawableSliderTail> { RelativeSizeAxes = Axes.Both },
tickContainer = new Container<DrawableSliderTick> { RelativeSizeAxes = Axes.Both },
repeatContainer = new Container<DrawableSliderRepeat> { RelativeSizeAxes = Axes.Both },
}
},
// slider head is not included in shake as it handles hit detection, and handles its own shaking.
headContainer = new Container<DrawableSliderHead> { RelativeSizeAxes = Axes.Both },
OverlayElementContainer = new Container { RelativeSizeAxes = Axes.Both, },
Ball,
2020-11-19 11:40:30 +00:00
slidingSample = new PausableSkinnableSound { Looping = true }
});
2018-04-13 09:19:50 +00:00
PositionBindable.BindValueChanged(_ => Position = HitObject.StackedPosition);
StackHeightBindable.BindValueChanged(_ => Position = HitObject.StackedPosition);
ScaleBindable.BindValueChanged(scale => Ball.Scale = new Vector2(scale.NewValue));
2018-04-13 09:19:50 +00:00
2019-07-22 05:45:25 +00:00
AccentColour.BindValueChanged(colour =>
{
2018-07-02 07:10:56 +00:00
foreach (var drawableHitObject in NestedHitObjects)
2019-07-22 05:45:25 +00:00
drawableHitObject.AccentColour.Value = colour.NewValue;
}, true);
2020-07-22 07:37:38 +00:00
Tracking.BindValueChanged(updateSlidingSample);
}
protected override void OnApply()
{
base.OnApply();
// Ensure that the version will change after the upcoming BindTo().
pathVersion.Value = int.MaxValue;
PathVersion.BindTo(HitObject.Path.Version);
}
public override void Shake() => shakeContainer.Shake();
protected override void OnFree()
{
base.OnFree();
PathVersion.UnbindFrom(HitObject.Path.Version);
slidingSample?.ClearSamples();
2020-11-19 11:40:30 +00:00
}
2020-07-22 07:37:38 +00:00
protected override void LoadSamples()
{
// Note: base.LoadSamples() isn't called since the slider plays the tail's hitsounds for the time being.
if (HitObject.SampleControlPoint == null)
{
throw new InvalidOperationException($"{nameof(HitObject)}s must always have an attached {nameof(HitObject.SampleControlPoint)}."
+ $" This is an indication that {nameof(HitObject.ApplyDefaults)} has not been invoked on {this}.");
}
Samples.Samples = HitObject.TailSamples.Select(s => HitObject.SampleControlPoint.ApplyTo(s)).Cast<ISampleInfo>().ToArray();
slidingSample.Samples = HitObject.CreateSlidingSamples().Select(s => HitObject.SampleControlPoint.ApplyTo(s)).Cast<ISampleInfo>().ToArray();
2020-07-22 07:37:38 +00:00
}
public override void StopAllSamples()
{
base.StopAllSamples();
slidingSample?.Stop();
}
2020-07-22 07:37:38 +00:00
private void updateSlidingSample(ValueChangedEvent<bool> tracking)
{
2020-09-29 03:45:20 +00:00
if (tracking.NewValue)
2020-07-22 07:37:38 +00:00
slidingSample?.Play();
else
slidingSample?.Stop();
}
2018-04-13 09:19:50 +00:00
protected override void AddNestedHitObject(DrawableHitObject hitObject)
{
base.AddNestedHitObject(hitObject);
2019-10-17 03:53:54 +00:00
switch (hitObject)
{
case DrawableSliderHead head:
headContainer.Child = head;
break;
case DrawableSliderTail tail:
tailContainer.Child = tail;
break;
case DrawableSliderTick tick:
tickContainer.Add(tick);
break;
case DrawableSliderRepeat repeat:
repeatContainer.Add(repeat);
break;
}
}
protected override void ClearNestedHitObjects()
{
base.ClearNestedHitObjects();
2020-11-12 06:59:48 +00:00
headContainer.Clear(false);
tailContainer.Clear(false);
repeatContainer.Clear(false);
tickContainer.Clear(false);
OverlayElementContainer.Clear(false);
}
protected override DrawableHitObject CreateNestedHitObject(HitObject hitObject)
{
switch (hitObject)
{
case SliderTailCircle tail:
2020-11-05 04:51:46 +00:00
return new DrawableSliderTail(tail);
case SliderHeadCircle head:
2020-11-12 06:59:48 +00:00
return new DrawableSliderHead(head);
case SliderTick tick:
2020-11-12 06:59:48 +00:00
return new DrawableSliderTick(tick);
2020-03-19 05:42:02 +00:00
case SliderRepeat repeat:
2020-11-12 06:59:48 +00:00
return new DrawableSliderRepeat(repeat);
}
return base.CreateNestedHitObject(hitObject);
}
public readonly Bindable<bool> Tracking = new Bindable<bool>();
2018-04-13 09:19:50 +00:00
protected override void Update()
2016-11-17 12:29:35 +00:00
{
base.Update();
2018-04-13 09:19:50 +00:00
Tracking.Value = Ball.Tracking;
2018-04-13 09:19:50 +00:00
2020-07-22 07:37:38 +00:00
if (Tracking.Value && slidingSample != null)
// keep the sliding sample playing at the current tracking position
slidingSample.Balance.Value = CalculateSamplePlaybackBalance(CalculateDrawableRelativePosition(Ball));
2020-07-22 07:37:38 +00:00
2020-11-05 04:51:46 +00:00
double completionProgress = Math.Clamp((Time.Current - HitObject.StartTime) / HitObject.Duration, 0, 1);
2018-04-13 09:19:50 +00:00
Ball.UpdateProgress(completionProgress);
SliderBody?.UpdateProgress(completionProgress);
foreach (DrawableHitObject hitObject in NestedHitObjects)
{
if (hitObject is ITrackSnaking s) s.UpdateSnakingPosition(HitObject.Path.PositionAt(SliderBody?.SnakedStart ?? 0), HitObject.Path.PositionAt(SliderBody?.SnakedEnd ?? 0));
if (hitObject is IRequireTracking t) t.Tracking = Ball.Tracking;
}
2018-04-13 09:19:50 +00:00
Size = SliderBody?.Size ?? Vector2.Zero;
OriginPosition = SliderBody?.PathOffset ?? Vector2.Zero;
2018-04-13 09:19:50 +00:00
2018-02-26 07:11:26 +00:00
if (DrawSize != Vector2.Zero)
{
var childAnchorPosition = Vector2.Divide(OriginPosition, DrawSize);
foreach (var obj in NestedHitObjects)
obj.RelativeAnchorPosition = childAnchorPosition;
Ball.RelativeAnchorPosition = childAnchorPosition;
}
}
2018-04-13 09:19:50 +00:00
public override void OnKilled()
{
base.OnKilled();
SliderBody?.RecyclePath();
}
protected override void CheckForResult(bool userTriggered, double timeOffset)
2016-11-29 12:40:24 +00:00
{
2020-11-05 04:51:46 +00:00
if (userTriggered || Time.Current < HitObject.EndTime)
return;
2021-02-10 09:52:39 +00:00
// If only the nested hitobjects are judged, then the slider's own judgement is ignored for scoring purposes.
// But the slider needs to still be judged with a reasonable hit/miss result for visual purposes (hit/miss transforms, etc).
if (HitObject.OnlyJudgeNestedObjects)
{
ApplyResult(r => r.Type = NestedHitObjects.Any(h => h.Result.IsHit) ? r.Judgement.MaxResult : r.Judgement.MinResult);
return;
}
2021-02-10 12:27:12 +00:00
// Otherwise, if this slider also needs to be judged, apply judgement proportionally to the number of nested hitobjects hit. This is the classic osu!stable scoring.
ApplyResult(r =>
{
int totalTicks = NestedHitObjects.Count;
int hitTicks = NestedHitObjects.Count(h => h.IsHit);
if (hitTicks == totalTicks)
r.Type = HitResult.Great;
2021-02-10 12:24:41 +00:00
else if (hitTicks == 0)
r.Type = HitResult.Miss;
2021-02-10 12:24:41 +00:00
else
{
2021-02-10 13:09:24 +00:00
double hitFraction = (double)hitTicks / totalTicks;
2021-02-10 12:25:31 +00:00
r.Type = hitFraction >= 0.5 ? HitResult.Ok : HitResult.Meh;
2021-02-10 12:24:41 +00:00
}
});
2020-03-26 10:51:02 +00:00
}
public override void PlaySamples()
{
// rather than doing it this way, we should probably attach the sample to the tail circle.
// this can only be done after we stop using LegacyLastTick.
if (!TailCircle.SamplePlaysOnlyOnHit || TailCircle.IsHit)
2020-03-26 10:51:02 +00:00
base.PlaySamples();
2016-11-29 12:40:24 +00:00
}
2018-04-13 09:19:50 +00:00
protected override void UpdateInitialTransforms()
{
base.UpdateInitialTransforms();
Body.FadeInFromZero(HitObject.TimeFadeIn);
}
2020-11-04 07:19:07 +00:00
protected override void UpdateStartTimeStateTransforms()
{
2020-11-04 07:19:07 +00:00
base.UpdateStartTimeStateTransforms();
2019-09-13 09:49:21 +00:00
Ball.FadeIn();
Ball.ScaleTo(HitObject.Scale);
2020-11-04 07:19:07 +00:00
}
2018-04-13 09:19:50 +00:00
2020-11-04 07:19:07 +00:00
protected override void UpdateHitStateTransforms(ArmedState state)
{
base.UpdateHitStateTransforms(state);
2018-04-13 09:19:50 +00:00
2022-10-18 20:43:31 +00:00
const float fade_out_time = 240;
2018-04-13 09:19:50 +00:00
2020-11-04 07:19:07 +00:00
switch (state)
{
case ArmedState.Hit:
if (SliderBody?.SnakingOut.Value == true)
Body.FadeOut(40); // short fade to allow for any body colour to smoothly disappear.
2020-11-04 07:19:07 +00:00
break;
}
2020-11-04 07:19:07 +00:00
2022-10-18 20:43:42 +00:00
this.FadeOut(fade_out_time).Expire();
}
2018-04-13 09:19:50 +00:00
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => SliderBody?.ReceivePositionalInputAt(screenSpacePos) ?? base.ReceivePositionalInputAt(screenSpacePos);
private partial class DefaultSliderBody : PlaySliderBody
{
}
}
}