2019-01-24 16:43:03 +08: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 17:19:50 +08:00
2022-06-17 15:37:17 +08:00
#nullable disable
2020-11-30 18:04:09 +08:00
using osu.Framework.Allocation ;
2016-09-02 19:30:27 +08:00
using osu.Framework.Graphics ;
2017-11-28 17:39:45 +08:00
using osu.Game.Beatmaps ;
2017-10-02 21:55:37 +08:00
using osu.Game.Rulesets.Catch.Objects ;
2020-02-19 17:01:59 +08:00
using osu.Game.Rulesets.Catch.Objects.Drawables ;
2017-09-06 16:02:13 +08:00
using osu.Game.Rulesets.Judgements ;
2017-08-08 11:46:37 +08:00
using osu.Game.Rulesets.Objects.Drawables ;
2018-01-04 18:22:15 +08:00
using osu.Game.Rulesets.UI.Scrolling ;
2019-11-29 17:25:11 +08:00
using osuTK ;
2018-04-13 17:19:50 +08:00
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Catch.UI
2016-09-02 19:30:27 +08:00
{
2017-09-12 17:19:28 +08:00
public class CatchPlayfield : ScrollingPlayfield
2016-09-02 19:30:27 +08:00
{
2020-07-01 23:21:45 +08:00
/// <summary>
/// The width of the playfield.
/// The horizontal movement of the catcher is confined in the area of this width.
/// </summary>
public const float WIDTH = 512 ;
2022-10-06 16:26:03 +08:00
/// <summary>
/// The height of the playfield.
/// This doesn't include the catcher area.
/// </summary>
public const float HEIGHT = 384 ;
2020-07-01 23:21:45 +08:00
/// <summary>
/// The center position of the playfield.
/// </summary>
public const float CENTER_X = WIDTH / 2 ;
2018-04-13 17:19:50 +08:00
2020-01-30 12:23:39 +08:00
public override bool ReceivePositionalInputAt ( Vector2 screenSpacePos ) = >
// only check the X position; handle all vertical space.
base . ReceivePositionalInputAt ( new Vector2 ( screenSpacePos . X , ScreenSpaceDrawQuad . Centre . Y ) ) ;
2019-11-29 17:25:11 +08:00
2021-07-19 19:18:17 +08:00
internal Catcher Catcher { get ; private set ; }
2021-07-19 18:44:40 +08:00
2021-07-19 19:18:17 +08:00
internal CatcherArea CatcherArea { get ; private set ; }
2021-07-19 18:44:40 +08:00
2021-10-01 13:56:42 +08:00
private readonly IBeatmapDifficultyInfo difficulty ;
2021-07-19 20:05:36 +08:00
2021-10-01 13:56:42 +08:00
public CatchPlayfield ( IBeatmapDifficultyInfo difficulty )
2021-07-19 19:18:17 +08:00
{
this . difficulty = difficulty ;
}
[BackgroundDependencyLoader]
private void load ( )
2016-09-02 19:30:27 +08:00
{
2021-07-19 19:11:49 +08:00
var droppedObjectContainer = new DroppedObjectContainer ( ) ;
2021-07-19 18:44:40 +08:00
2021-07-27 17:59:55 +08:00
Catcher = new Catcher ( droppedObjectContainer , difficulty )
2020-07-15 19:58:09 +08:00
{
2021-07-19 18:44:40 +08:00
X = CENTER_X
2020-07-15 19:58:09 +08:00
} ;
2018-04-13 17:19:50 +08:00
2021-07-19 19:18:17 +08:00
AddRangeInternal ( new [ ]
2017-08-02 19:28:24 +08:00
{
2021-07-19 19:11:49 +08:00
droppedObjectContainer ,
2021-07-19 18:44:40 +08:00
Catcher . CreateProxiedContent ( ) ,
2021-04-09 15:04:45 +08:00
HitObjectContainer . CreateProxy ( ) ,
// This ordering (`CatcherArea` before `HitObjectContainer`) is important to
// make sure the up-to-date catcher position is used for the catcher catching logic of hit objects.
2021-07-19 18:44:40 +08:00
CatcherArea = new CatcherArea
{
Anchor = Anchor . BottomLeft ,
Origin = Anchor . TopLeft ,
2021-07-21 15:28:31 +08:00
Catcher = Catcher ,
2021-07-19 18:44:40 +08:00
} ,
2021-04-09 15:04:45 +08:00
HitObjectContainer ,
2021-07-19 19:18:17 +08:00
} ) ;
2020-11-30 18:04:09 +08:00
2020-12-01 13:50:42 +08:00
RegisterPool < Droplet , DrawableDroplet > ( 50 ) ;
RegisterPool < TinyDroplet , DrawableTinyDroplet > ( 50 ) ;
RegisterPool < Fruit , DrawableFruit > ( 100 ) ;
RegisterPool < Banana , DrawableBanana > ( 100 ) ;
RegisterPool < JuiceStream , DrawableJuiceStream > ( 10 ) ;
RegisterPool < BananaShower , DrawableBananaShower > ( 2 ) ;
2020-11-30 18:04:09 +08:00
}
2020-11-30 14:20:52 +08:00
protected override void LoadComplete ( )
{
base . LoadComplete ( ) ;
2020-11-20 16:25:57 +08:00
2020-11-30 14:22:55 +08:00
// these subscriptions need to be done post constructor to ensure externally bound components have a chance to populate required fields (ScoreProcessor / ComboAtJudgement in this case).
2020-11-20 16:25:57 +08:00
NewResult + = onNewResult ;
RevertResult + = onRevertResult ;
2020-11-21 14:20:33 +08:00
}
protected override void OnNewDrawableHitObject ( DrawableHitObject d )
{
2020-11-22 17:36:10 +08:00
( ( DrawableCatchHitObject ) d ) . CheckPosition = checkIfWeCanCatch ;
2016-09-02 19:30:27 +08:00
}
2018-04-13 17:19:50 +08:00
2021-07-19 18:52:40 +08:00
private bool checkIfWeCanCatch ( CatchHitObject obj ) = > Catcher . CanCatch ( obj ) ;
2018-04-13 17:19:50 +08:00
2018-08-06 09:54:16 +08:00
private void onNewResult ( DrawableHitObject judgedObject , JudgementResult result )
2020-09-13 04:39:06 +08:00
= > CatcherArea . OnNewResult ( ( DrawableCatchHitObject ) judgedObject , result ) ;
2020-08-04 03:13:02 +08:00
private void onRevertResult ( DrawableHitObject judgedObject , JudgementResult result )
2020-09-13 04:39:06 +08:00
= > CatcherArea . OnRevertResult ( ( DrawableCatchHitObject ) judgedObject , result ) ;
2016-09-02 19:30:27 +08:00
}
2017-08-08 09:35:56 +08:00
}