2019-01-24 08:43:03 +00:00
|
|
|
|
// 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
|
|
|
|
|
2021-08-01 13:03:51 +00:00
|
|
|
|
using System;
|
2020-07-04 07:45:46 +00:00
|
|
|
|
using System.Diagnostics;
|
2017-03-23 03:49:28 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Graphics;
|
2020-11-20 07:14:38 +00:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
2020-07-04 07:45:46 +00:00
|
|
|
|
using osu.Framework.Graphics.Pooling;
|
2018-03-07 10:14:42 +00:00
|
|
|
|
using osu.Game.Rulesets.Objects.Drawables;
|
2017-12-30 20:23:18 +00:00
|
|
|
|
using osu.Game.Rulesets.Scoring;
|
2018-03-07 09:20:35 +00:00
|
|
|
|
using osu.Game.Skinning;
|
2020-11-17 05:59:34 +00:00
|
|
|
|
using osuTK;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2017-04-18 07:05:58 +00:00
|
|
|
|
namespace osu.Game.Rulesets.Judgements
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
2017-03-23 10:00:18 +00:00
|
|
|
|
/// A drawable object which visualises the hit result of a <see cref="Judgements.Judgement"/>.
|
2017-03-23 03:49:28 +00:00
|
|
|
|
/// </summary>
|
2020-07-04 07:45:46 +00:00
|
|
|
|
public partial class DrawableJudgement : PoolableDrawable
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
2019-09-17 17:56:03 +00:00
|
|
|
|
private const float judgement_size = 128;
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2024-05-29 09:11:43 +00:00
|
|
|
|
public JudgementResult? Result { get; private set; }
|
2020-11-17 05:59:34 +00:00
|
|
|
|
|
2024-05-29 09:11:43 +00:00
|
|
|
|
public DrawableHitObject? JudgedObject { get; private set; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-11-18 08:53:27 +00:00
|
|
|
|
public override bool RemoveCompletedTransforms => false;
|
|
|
|
|
|
2024-05-29 09:11:43 +00:00
|
|
|
|
protected SkinnableDrawable? JudgementBody { get; private set; }
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-11-20 07:14:38 +00:00
|
|
|
|
private readonly Container aboveHitObjectsContent;
|
|
|
|
|
|
2021-08-01 13:03:51 +00:00
|
|
|
|
private readonly Lazy<Drawable> proxiedAboveHitObjectsContent;
|
|
|
|
|
public Drawable ProxiedAboveHitObjectsContent => proxiedAboveHitObjectsContent.Value;
|
|
|
|
|
|
2017-03-23 03:49:28 +00:00
|
|
|
|
/// <summary>
|
2017-03-23 10:00:18 +00:00
|
|
|
|
/// Creates a drawable which visualises a <see cref="Judgements.Judgement"/>.
|
2017-03-23 03:49:28 +00:00
|
|
|
|
/// </summary>
|
2018-08-02 11:35:54 +00:00
|
|
|
|
/// <param name="result">The judgement to visualise.</param>
|
2018-03-24 09:22:55 +00:00
|
|
|
|
/// <param name="judgedObject">The object which was judged.</param>
|
2018-08-02 11:35:54 +00:00
|
|
|
|
public DrawableJudgement(JudgementResult result, DrawableHitObject judgedObject)
|
2020-07-04 07:45:46 +00:00
|
|
|
|
: this()
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
2020-07-10 09:34:31 +00:00
|
|
|
|
Apply(result, judgedObject);
|
2017-03-23 03:49:28 +00:00
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-07-04 07:45:46 +00:00
|
|
|
|
public DrawableJudgement()
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
2020-07-04 07:45:46 +00:00
|
|
|
|
Size = new Vector2(judgement_size);
|
2020-07-06 03:54:39 +00:00
|
|
|
|
Origin = Anchor.Centre;
|
2020-11-20 07:14:38 +00:00
|
|
|
|
|
|
|
|
|
AddInternal(aboveHitObjectsContent = new Container
|
|
|
|
|
{
|
|
|
|
|
Depth = float.MinValue,
|
|
|
|
|
RelativeSizeAxes = Axes.Both
|
|
|
|
|
});
|
2021-08-01 13:03:51 +00:00
|
|
|
|
|
|
|
|
|
proxiedAboveHitObjectsContent = new Lazy<Drawable>(() => aboveHitObjectsContent.CreateProxy());
|
2017-03-23 03:49:28 +00:00
|
|
|
|
}
|
2018-04-13 09:19:50 +00:00
|
|
|
|
|
2020-07-10 09:34:31 +00:00
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
|
|
|
|
{
|
|
|
|
|
prepareDrawables();
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-17 05:59:34 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Apply top-level animations to the current judgement when successfully hit.
|
2020-11-18 10:38:38 +00:00
|
|
|
|
/// If displaying components which require lifetime extensions, manually adjusting <see cref="Drawable.LifetimeEnd"/> is required.
|
2020-11-17 05:59:34 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// For animating the actual "default skin" judgement itself, it is recommended to use <see cref="CreateDefaultJudgement"/>.
|
|
|
|
|
/// This allows applying animations which don't affect custom skins.
|
|
|
|
|
/// </remarks>
|
2019-03-12 10:41:33 +00:00
|
|
|
|
protected virtual void ApplyHitAnimations()
|
|
|
|
|
{
|
|
|
|
|
}
|
2019-03-12 10:23:24 +00:00
|
|
|
|
|
2020-11-17 05:59:34 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Apply top-level animations to the current judgement when missed.
|
2020-11-18 10:38:38 +00:00
|
|
|
|
/// If displaying components which require lifetime extensions, manually adjusting <see cref="Drawable.LifetimeEnd"/> is required.
|
2020-11-17 05:59:34 +00:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <remarks>
|
|
|
|
|
/// For animating the actual "default skin" judgement itself, it is recommended to use <see cref="CreateDefaultJudgement"/>.
|
|
|
|
|
/// This allows applying animations which don't affect custom skins.
|
|
|
|
|
/// </remarks>
|
|
|
|
|
protected virtual void ApplyMissAnimations()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-17 06:03:26 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Associate a new result / object with this judgement. Should be called when retrieving a judgement from a pool.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="result">The applicable judgement.</param>
|
|
|
|
|
/// <param name="judgedObject">The drawable object.</param>
|
2024-05-29 09:11:43 +00:00
|
|
|
|
public void Apply(JudgementResult result, DrawableHitObject? judgedObject)
|
2020-07-04 07:45:46 +00:00
|
|
|
|
{
|
2020-07-10 09:34:31 +00:00
|
|
|
|
Result = result;
|
|
|
|
|
JudgedObject = judgedObject;
|
2020-07-04 07:45:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-29 09:11:43 +00:00
|
|
|
|
protected override void FreeAfterUse()
|
|
|
|
|
{
|
|
|
|
|
base.FreeAfterUse();
|
|
|
|
|
|
|
|
|
|
JudgedObject = null;
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-04 07:45:46 +00:00
|
|
|
|
protected override void PrepareForUse()
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
2020-07-04 07:45:46 +00:00
|
|
|
|
base.PrepareForUse();
|
|
|
|
|
|
|
|
|
|
Debug.Assert(Result != null);
|
|
|
|
|
|
2020-11-18 08:53:27 +00:00
|
|
|
|
runAnimation();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void runAnimation()
|
|
|
|
|
{
|
2020-11-24 06:19:51 +00:00
|
|
|
|
// is a no-op if the drawables are already in a correct state.
|
|
|
|
|
prepareDrawables();
|
|
|
|
|
|
2020-11-23 06:18:54 +00:00
|
|
|
|
// undo any transforms applies in ApplyMissAnimations/ApplyHitAnimations to get a sane initial state.
|
2020-11-25 03:11:44 +00:00
|
|
|
|
ApplyTransformsAt(double.MinValue, true);
|
|
|
|
|
ClearTransforms(true);
|
2020-11-23 06:18:54 +00:00
|
|
|
|
|
2024-05-29 09:11:43 +00:00
|
|
|
|
Debug.Assert(Result != null && JudgementBody != null);
|
|
|
|
|
|
2020-11-18 06:51:09 +00:00
|
|
|
|
LifetimeStart = Result.TimeAbsolute;
|
2020-07-13 11:12:50 +00:00
|
|
|
|
|
2021-07-05 15:52:39 +00:00
|
|
|
|
using (BeginAbsoluteSequence(Result.TimeAbsolute))
|
2017-03-23 03:49:28 +00:00
|
|
|
|
{
|
2020-11-18 06:51:09 +00:00
|
|
|
|
// not sure if this should remain going forward.
|
2020-11-18 10:34:27 +00:00
|
|
|
|
JudgementBody.ResetAnimation();
|
2019-04-01 03:44:46 +00:00
|
|
|
|
|
2020-11-18 06:51:09 +00:00
|
|
|
|
switch (Result.Type)
|
|
|
|
|
{
|
|
|
|
|
case HitResult.None:
|
|
|
|
|
break;
|
2019-04-01 03:44:46 +00:00
|
|
|
|
|
2020-11-18 06:51:09 +00:00
|
|
|
|
default:
|
2023-12-20 11:23:19 +00:00
|
|
|
|
if (Result.Type.IsHit())
|
|
|
|
|
ApplyHitAnimations();
|
|
|
|
|
else
|
|
|
|
|
ApplyMissAnimations();
|
2020-11-18 06:51:09 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-18 10:34:27 +00:00
|
|
|
|
if (JudgementBody.Drawable is IAnimatableJudgement animatable)
|
2020-11-17 06:43:54 +00:00
|
|
|
|
animatable.PlayAnimation();
|
2020-11-17 05:59:34 +00:00
|
|
|
|
|
2021-03-06 15:02:20 +00:00
|
|
|
|
// a derived version of DrawableJudgement may be proposing a lifetime.
|
|
|
|
|
// if not adjusted (or the skinned portion requires greater bounds than calculated) use the skinned source's lifetime.
|
|
|
|
|
double lastTransformTime = JudgementBody.Drawable.LatestTransformEndTime;
|
|
|
|
|
if (LifetimeEnd == double.MaxValue || lastTransformTime > LifetimeEnd)
|
|
|
|
|
LifetimeEnd = lastTransformTime;
|
2020-11-18 06:51:09 +00:00
|
|
|
|
}
|
2017-03-23 03:49:28 +00:00
|
|
|
|
}
|
2020-07-04 07:45:46 +00:00
|
|
|
|
|
2020-07-10 09:34:31 +00:00
|
|
|
|
private HitResult? currentDrawableType;
|
|
|
|
|
|
2020-07-04 07:45:46 +00:00
|
|
|
|
private void prepareDrawables()
|
|
|
|
|
{
|
|
|
|
|
var type = Result?.Type ?? HitResult.Perfect; //TODO: better default type from ruleset
|
|
|
|
|
|
2020-11-17 06:43:54 +00:00
|
|
|
|
// todo: this should be removed once judgements are always pooled.
|
2020-07-10 09:34:31 +00:00
|
|
|
|
if (type == currentDrawableType)
|
|
|
|
|
return;
|
|
|
|
|
|
2020-07-26 21:22:31 +00:00
|
|
|
|
// sub-classes might have added their own children that would be removed here if .InternalChild was used.
|
|
|
|
|
if (JudgementBody != null)
|
2022-08-29 06:49:28 +00:00
|
|
|
|
RemoveInternal(JudgementBody, true);
|
2020-07-26 21:22:31 +00:00
|
|
|
|
|
2022-11-09 07:04:56 +00:00
|
|
|
|
AddInternal(JudgementBody = new SkinnableDrawable(new GameplaySkinComponentLookup<HitResult>(type), _ =>
|
2022-11-07 08:05:16 +00:00
|
|
|
|
CreateDefaultJudgement(type), confineMode: ConfineMode.NoScaling));
|
2020-07-04 07:45:46 +00:00
|
|
|
|
|
2020-11-24 06:19:51 +00:00
|
|
|
|
JudgementBody.OnSkinChanged += () =>
|
2020-11-20 07:14:38 +00:00
|
|
|
|
{
|
2020-11-24 06:19:51 +00:00
|
|
|
|
// on a skin change, the child component will update but not get correctly triggered to play its animation (or proxy the newly created content).
|
|
|
|
|
// we need to trigger a reinitialisation to make things right.
|
|
|
|
|
proxyContent();
|
|
|
|
|
runAnimation();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proxyContent();
|
2020-11-20 07:14:38 +00:00
|
|
|
|
|
2020-07-10 09:34:31 +00:00
|
|
|
|
currentDrawableType = type;
|
2020-11-24 06:19:51 +00:00
|
|
|
|
|
|
|
|
|
void proxyContent()
|
|
|
|
|
{
|
|
|
|
|
aboveHitObjectsContent.Clear();
|
|
|
|
|
|
|
|
|
|
if (JudgementBody.Drawable is IAnimatableJudgement animatable)
|
|
|
|
|
{
|
|
|
|
|
var proxiedContent = animatable.GetAboveHitObjectsProxiedContent();
|
|
|
|
|
if (proxiedContent != null)
|
|
|
|
|
aboveHitObjectsContent.Add(proxiedContent);
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-07-04 07:45:46 +00:00
|
|
|
|
}
|
2020-11-17 05:59:34 +00:00
|
|
|
|
|
2020-11-17 06:43:54 +00:00
|
|
|
|
protected virtual Drawable CreateDefaultJudgement(HitResult result) => new DefaultJudgementPiece(result);
|
2017-03-23 03:49:28 +00:00
|
|
|
|
}
|
|
|
|
|
}
|