osu/osu.Game.Rulesets.Taiko/UI/TaikoPlayfield.cs

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

356 lines
14 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
using System;
2021-03-06 14:06:16 +00:00
using System.Collections.Generic;
using System.Linq;
2016-11-14 08:23:33 +00:00
using osu.Framework.Allocation;
2016-09-02 10:58:57 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2017-03-21 06:09:54 +00:00
using osu.Game.Graphics;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Judgements;
using osu.Game.Rulesets.Objects;
2020-09-25 10:25:58 +00:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Rulesets.UI;
2018-01-04 10:22:15 +00:00
using osu.Game.Rulesets.UI.Scrolling;
using osu.Game.Rulesets.Taiko.Objects.Drawables;
using osu.Game.Rulesets.Taiko.Judgements;
using osu.Game.Rulesets.Taiko.Objects;
2021-03-06 14:06:16 +00:00
using osu.Game.Rulesets.Taiko.Scoring;
2020-04-21 10:00:34 +00:00
using osu.Game.Skinning;
2018-04-13 09:19:50 +00:00
2017-04-18 07:05:58 +00:00
namespace osu.Game.Rulesets.Taiko.UI
2016-09-02 10:58:57 +00:00
{
public partial class TaikoPlayfield : ScrollingPlayfield
2016-09-02 10:58:57 +00:00
{
2017-03-21 06:09:54 +00:00
/// <summary>
/// Base height of a <see cref="TaikoPlayfield"/> when inside a <see cref="DrawableTaikoRuleset"/>.
2017-03-21 06:09:54 +00:00
/// </summary>
public const float BASE_HEIGHT = 200;
2018-04-13 09:19:50 +00:00
public const float INPUT_DRUM_WIDTH = 180f;
public Container UnderlayElements { get; private set; } = null!;
private Container<HitExplosion> hitExplosionContainer = null!;
private Container<KiaiHitExplosion> kiaiExplosionContainer = null!;
private JudgementContainer<DrawableTaikoJudgement> judgementContainer = null!;
private ScrollingHitObjectContainer drumRollHitContainer = null!;
internal Drawable HitTarget = null!;
2018-04-13 09:19:50 +00:00
private JudgementPooler<DrawableTaikoJudgement> judgementPooler = null!;
private readonly IDictionary<HitResult, HitExplosionPool> explosionPools = new Dictionary<HitResult, HitExplosionPool>();
2021-03-06 14:06:16 +00:00
private ProxyContainer topLevelHitContainer = null!;
private InputDrum inputDrum = null!;
2018-04-13 09:19:50 +00:00
/// <remarks>
/// <see cref="Playfield.AddNested"/> is purposefully not called on this to prevent i.e. being able to interact
/// with bar lines in the editor.
/// </remarks>
private BarLinePlayfield barLinePlayfield = null!;
[BackgroundDependencyLoader]
private void load(OsuColour colours)
2016-09-02 10:58:57 +00:00
{
const float hit_target_width = BASE_HEIGHT;
const float hit_target_offset = -24f;
2022-07-22 07:18:22 +00:00
inputDrum = new InputDrum
2022-04-13 01:38:07 +00:00
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
RelativeSizeAxes = Axes.Y,
Width = INPUT_DRUM_WIDTH,
2022-04-13 01:38:07 +00:00
};
2020-04-25 05:47:20 +00:00
InternalChildren = new[]
2017-03-21 06:09:54 +00:00
{
new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.PlayfieldBackgroundRight), _ => new PlayfieldBackgroundRight()),
2022-04-13 01:38:07 +00:00
new Container
{
Name = "Left overlay",
RelativeSizeAxes = Axes.Y,
Width = INPUT_DRUM_WIDTH,
BorderColour = colours.Gray0,
2022-04-13 01:38:07 +00:00
Children = new[]
{
new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.PlayfieldBackgroundLeft), _ => new PlayfieldBackgroundLeft()),
2022-04-13 01:38:07 +00:00
inputDrum.CreateProxy(),
}
},
new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.Mascot), _ => Empty())
{
Origin = Anchor.BottomLeft,
Anchor = Anchor.TopLeft,
RelativePositionAxes = Axes.Y,
RelativeSizeAxes = Axes.None,
Y = 0.2f
},
new Container
{
Name = "Right area",
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding { Left = INPUT_DRUM_WIDTH },
Children = new Drawable[]
2017-08-02 22:46:19 +00:00
{
new Container
2017-03-21 06:09:54 +00:00
{
Name = "Elements behind hit objects",
RelativeSizeAxes = Axes.Y,
Width = hit_target_width,
X = hit_target_offset,
2020-04-21 10:00:34 +00:00
Children = new[]
2017-03-21 06:09:54 +00:00
{
2022-11-29 23:44:20 +00:00
new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.KiaiGlow), _ => Empty())
{
RelativeSizeAxes = Axes.Both,
},
hitExplosionContainer = new Container<HitExplosion>
{
RelativeSizeAxes = Axes.Both,
},
HitTarget = new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.HitTarget), _ => new TaikoHitTarget())
2017-03-21 06:09:54 +00:00
{
RelativeSizeAxes = Axes.Both,
}
}
},
new Container
{
Name = "Bar line content",
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding { Left = hit_target_width / 2 + hit_target_offset },
Children = new Drawable[]
{
UnderlayElements = new Container
{
RelativeSizeAxes = Axes.Both,
},
barLinePlayfield = new BarLinePlayfield(),
}
},
new Container
{
Name = "Masked hit objects content",
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding { Left = hit_target_width / 2 + hit_target_offset },
Masking = true,
Child = HitObjectContainer,
},
new Container
{
Name = "Overlay content",
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding { Left = hit_target_width / 2 + hit_target_offset },
Children = new Drawable[]
{
drumRollHitContainer = new DrumRollHitContainer(),
kiaiExplosionContainer = new Container<KiaiHitExplosion>
{
Name = "Kiai hit explosions",
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.Both,
FillMode = FillMode.Fit,
},
judgementContainer = new JudgementContainer<DrawableTaikoJudgement>
{
Name = "Judgements",
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.Both,
FillMode = FillMode.Fit,
},
}
},
}
},
topLevelHitContainer = new ProxyContainer
{
Name = "Top level hit objects",
RelativeSizeAxes = Axes.Both,
2020-04-22 13:19:29 +00:00
},
drumRollHitContainer.CreateProxy(),
new SkinnableDrawable(new TaikoSkinComponentLookup(TaikoSkinComponents.DrumSamplePlayer), _ => new DrumSamplePlayer())
{
RelativeSizeAxes = Axes.Both,
},
2022-04-13 01:38:07 +00:00
// this is added at the end of the hierarchy to receive input before taiko objects.
// but is proxied below everything to not cover visual effects such as hit explosions.
inputDrum,
};
2020-12-20 17:16:05 +00:00
RegisterPool<Hit, DrawableHit>(50);
RegisterPool<Hit.StrongNestedHit, DrawableHit.StrongNestedHit>(50);
RegisterPool<DrumRoll, DrawableDrumRoll>(5);
RegisterPool<DrumRoll.StrongNestedHit, DrawableDrumRoll.StrongNestedHit>(5);
RegisterPool<DrumRollTick, DrawableDrumRollTick>(100);
RegisterPool<DrumRollTick.StrongNestedHit, DrawableDrumRollTick.StrongNestedHit>(100);
RegisterPool<Swell, DrawableSwell>(5);
RegisterPool<SwellTick, DrawableSwellTick>(100);
2021-03-06 14:06:16 +00:00
var hitWindows = new TaikoHitWindows();
HitResult[] usableHitResults = Enum.GetValues<HitResult>().Where(r => hitWindows.IsHitResultAllowed(r)).ToArray();
2021-03-06 14:06:16 +00:00
AddInternal(judgementPooler = new JudgementPooler<DrawableTaikoJudgement>(usableHitResults));
foreach (var result in usableHitResults)
explosionPools.Add(result, new HitExplosionPool(result));
AddRangeInternal(explosionPools.Values);
2016-09-02 10:58:57 +00:00
}
2018-04-13 09:19:50 +00:00
protected override void LoadComplete()
{
base.LoadComplete();
NewResult += OnNewResult;
}
protected override void OnNewDrawableHitObject(DrawableHitObject drawableHitObject)
{
base.OnNewDrawableHitObject(drawableHitObject);
var taikoObject = (DrawableTaikoHitObject)drawableHitObject;
topLevelHitContainer.Add(taikoObject.CreateProxiedContent());
}
#region Pooling support
public override void Add(HitObject h)
{
switch (h)
{
case BarLine barLine:
barLinePlayfield.Add(barLine);
break;
case TaikoHitObject taikoHitObject:
base.Add(taikoHitObject);
break;
default:
throw new ArgumentException($"Unsupported {nameof(HitObject)} type: {h.GetType()}");
}
}
public override bool Remove(HitObject h)
{
switch (h)
{
case BarLine barLine:
return barLinePlayfield.Remove(barLine);
case TaikoHitObject taikoHitObject:
return base.Remove(taikoHitObject);
default:
throw new ArgumentException($"Unsupported {nameof(HitObject)} type: {h.GetType()}");
}
}
#endregion
#region Non-pooling support
public override void Add(DrawableHitObject h)
2017-03-21 06:09:54 +00:00
{
switch (h)
{
case DrawableBarLine barLine:
barLinePlayfield.Add(barLine);
break;
2019-04-01 03:44:46 +00:00
2022-06-24 12:25:23 +00:00
case DrawableTaikoHitObject:
base.Add(h);
break;
default:
throw new ArgumentException($"Unsupported {nameof(DrawableHitObject)} type: {h.GetType()}");
}
}
public override bool Remove(DrawableHitObject h)
{
switch (h)
{
case DrawableBarLine barLine:
return barLinePlayfield.Remove(barLine);
2022-06-24 12:25:23 +00:00
case DrawableTaikoHitObject:
return base.Remove(h);
default:
throw new ArgumentException($"Unsupported {nameof(DrawableHitObject)} type: {h.GetType()}");
}
2017-03-21 06:09:54 +00:00
}
2018-04-13 09:19:50 +00:00
#endregion
internal void OnNewResult(DrawableHitObject judgedObject, JudgementResult result)
2017-03-21 06:09:54 +00:00
{
2019-02-21 09:56:34 +00:00
if (!DisplayJudgements.Value)
return;
if (!judgedObject.DisplayResult)
return;
2018-08-03 07:46:03 +00:00
switch (result.Judgement)
2017-03-21 07:33:25 +00:00
{
2022-06-24 12:25:23 +00:00
case TaikoStrongJudgement:
2018-08-03 07:46:03 +00:00
if (result.IsHit)
hitExplosionContainer.Children.FirstOrDefault(e => e.JudgedObject == ((DrawableStrongNestedHit)judgedObject).ParentHitObject)?.VisualiseSecondHit(result);
2018-08-03 07:46:03 +00:00
break;
2019-04-01 03:44:46 +00:00
2022-06-24 12:25:23 +00:00
case TaikoDrumRollTickJudgement:
2020-04-27 03:23:53 +00:00
if (!result.IsHit)
break;
2020-04-27 03:23:53 +00:00
var drawableTick = (DrawableDrumRollTick)judgedObject;
addDrumRollHit(drawableTick);
2020-04-27 03:01:31 +00:00
break;
2018-08-03 07:46:03 +00:00
default:
2022-09-06 08:27:29 +00:00
if (!result.Type.IsScorable())
break;
var judgement = judgementPooler.Get(result.Type, j => j.Apply(result, judgedObject));
if (judgement == null)
return;
judgementContainer.Add(judgement);
2018-04-13 09:19:50 +00:00
var type = (judgedObject.HitObject as Hit)?.Type ?? HitType.Centre;
2020-09-25 10:25:58 +00:00
addExplosion(judgedObject, result.Type, type);
2018-08-03 07:46:03 +00:00
break;
}
2020-04-22 13:19:29 +00:00
}
private void addDrumRollHit(DrawableDrumRollTick drawableTick) =>
drumRollHitContainer.Add(new DrawableFlyingHit(drawableTick));
2020-04-27 03:01:31 +00:00
private void addExplosion(DrawableHitObject drawableObject, HitResult result, HitType type)
2020-04-27 03:23:53 +00:00
{
hitExplosionContainer.Add(explosionPools[result]
.Get(explosion => explosion.Apply(drawableObject)));
2020-04-27 03:23:53 +00:00
if (drawableObject.HitObject.Kiai)
kiaiExplosionContainer.Add(new KiaiHitExplosion(drawableObject, type));
}
2020-04-27 03:23:53 +00:00
2020-04-26 23:40:57 +00:00
private partial class ProxyContainer : LifetimeManagementContainer
2020-04-22 13:19:29 +00:00
{
public void Add(Drawable proxy) => AddInternal(proxy);
public override bool UpdateSubTreeMasking()
2020-04-26 23:40:57 +00:00
{
// DrawableHitObject disables masking.
// Hitobject content is proxied and unproxied based on hit status and the IsMaskedAway value could get stuck because of this.
return false;
2020-04-26 23:40:57 +00:00
}
}
2016-09-02 10:58:57 +00:00
}
}