1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-30 20:12:56 +08:00
osu-lazer/osu.Game.Rulesets.Catch/UI/CatcherArea.cs

188 lines
6.3 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 osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Input.Bindings;
2018-06-29 15:49:01 +08:00
using osu.Game.Rulesets.Catch.Judgements;
using osu.Game.Rulesets.Catch.Objects.Drawables;
2018-04-13 17:19:50 +08:00
using osu.Game.Rulesets.Catch.Replays;
using osu.Game.Rulesets.Judgements;
2020-09-29 13:26:36 +08:00
using osu.Game.Rulesets.Scoring;
2018-06-11 22:00:26 +08:00
using osu.Game.Rulesets.UI;
2018-11-20 15:51:59 +08:00
using osuTK;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Rulesets.Catch.UI
{
2021-07-21 15:40:35 +08:00
/// <summary>
/// The horizontal band at the bottom of the playfield the catcher is moving on.
/// It holds a <see cref="Catcher"/> as a child and translates input to the catcher movement.
/// It also holds a combo display that is above the catcher, and judgment results are translated to the catcher and the combo display.
/// </summary>
public class CatcherArea : Container, IKeyBindingHandler<CatchAction>
2018-04-13 17:19:50 +08:00
{
public Catcher Catcher
2021-07-19 18:44:40 +08:00
{
get => catcher;
set => catcherContainer.Child = catcher = value;
2021-07-19 18:44:40 +08:00
}
private readonly Container<Catcher> catcherContainer;
2020-09-13 04:39:06 +08:00
private readonly CatchComboDisplay comboDisplay;
private readonly CatcherTrailDisplay catcherTrails;
2021-07-19 18:44:40 +08:00
private Catcher catcher;
/// <summary>
/// <c>-1</c> when only left button is pressed.
/// <c>1</c> when only right button is pressed.
/// <c>0</c> when none or both left and right buttons are pressed.
/// </summary>
private int currentDirection;
// TODO: support replay rewind
private bool lastHyperDashState;
2021-07-21 15:40:35 +08:00
/// <remarks>
/// <see cref="Catcher"/> must be set before loading.
/// </remarks>
2021-07-19 18:44:40 +08:00
public CatcherArea()
2018-04-13 17:19:50 +08:00
{
Size = new Vector2(CatchPlayfield.WIDTH, Catcher.BASE_SIZE);
Children = new Drawable[]
{
catcherContainer = new Container<Catcher> { RelativeSizeAxes = Axes.Both },
catcherTrails = new CatcherTrailDisplay(),
comboDisplay = new CatchComboDisplay
{
RelativeSizeAxes = Axes.None,
AutoSizeAxes = Axes.Both,
Anchor = Anchor.TopLeft,
Origin = Anchor.Centre,
Margin = new MarginPadding { Bottom = 350f },
X = CatchPlayfield.CENTER_X
}
};
}
2018-04-13 17:19:50 +08:00
public void OnNewResult(DrawableCatchHitObject hitObject, JudgementResult result)
2018-04-13 17:19:50 +08:00
{
Catcher.OnNewResult(hitObject, result);
2020-09-29 13:26:36 +08:00
if (!result.Type.IsScorable())
return;
if (hitObject.HitObject.LastInCombo)
2018-04-13 17:19:50 +08:00
{
if (result.Judgement is CatchJudgement catchJudgement && catchJudgement.ShouldExplodeFor(result))
Catcher.Explode();
2018-04-13 17:19:50 +08:00
else
Catcher.Drop();
2018-04-13 17:19:50 +08:00
}
2020-09-13 04:39:06 +08:00
comboDisplay.OnNewResult(hitObject, result);
2018-04-13 17:19:50 +08:00
}
public void OnRevertResult(DrawableCatchHitObject hitObject, JudgementResult result)
{
comboDisplay.OnRevertResult(hitObject, result);
Catcher.OnRevertResult(hitObject, result);
2018-04-13 17:19:50 +08:00
}
protected override void Update()
{
base.Update();
var replayState = (GetContainingInputManager().CurrentState as RulesetInputManagerInputState<CatchAction>)?.LastReplayState as CatchFramedReplayInputHandler.CatchReplayState;
SetCatcherPosition(
replayState?.CatcherX ??
(float)(Catcher.X + Catcher.Speed * currentDirection * Clock.ElapsedFrameTime));
}
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
comboDisplay.X = Catcher.X;
if (Time.Elapsed <= 0)
{
// This is probably a wrong value, but currently the true value is not recorded.
2021-07-30 14:44:09 +08:00
// Setting `true` will prevent generation of false-positive after-images (with more false-negatives).
lastHyperDashState = true;
return;
}
if (!lastHyperDashState && Catcher.HyperDashing)
2021-08-02 14:08:42 +08:00
displayCatcherTrail(CatcherTrailAnimation.HyperDashAfterImage);
if (Catcher.Dashing || Catcher.HyperDashing)
{
double generationInterval = Catcher.HyperDashing ? 25 : 50;
if (Time.Current - catcherTrails.LastDashTrailTime >= generationInterval)
displayCatcherTrail(Catcher.HyperDashing ? CatcherTrailAnimation.HyperDashing : CatcherTrailAnimation.Dashing);
}
lastHyperDashState = Catcher.HyperDashing;
}
public void SetCatcherPosition(float X)
{
float lastPosition = Catcher.X;
float newPosition = Math.Clamp(X, 0, CatchPlayfield.WIDTH);
Catcher.X = newPosition;
if (lastPosition < newPosition)
Catcher.VisualDirection = Direction.Right;
else if (lastPosition > newPosition)
Catcher.VisualDirection = Direction.Left;
}
public bool OnPressed(CatchAction action)
{
switch (action)
{
case CatchAction.MoveLeft:
currentDirection--;
return true;
case CatchAction.MoveRight:
currentDirection++;
return true;
case CatchAction.Dash:
Catcher.Dashing = true;
return true;
}
return false;
}
public void OnReleased(CatchAction action)
{
switch (action)
{
case CatchAction.MoveLeft:
currentDirection++;
break;
case CatchAction.MoveRight:
currentDirection--;
break;
case CatchAction.Dash:
Catcher.Dashing = false;
break;
}
}
private void displayCatcherTrail(CatcherTrailAnimation animation) => catcherTrails.Add(new CatcherTrailEntry(Time.Current, Catcher.CurrentState, Catcher.X, Catcher.BodyScale, animation));
}
2018-04-13 17:19:50 +08:00
}