osu/osu.Game.Rulesets.Mania/UI/Column.cs

182 lines
6.2 KiB
C#
Raw Normal View History

2018-04-13 09:19:50 +00:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System.Linq;
2018-11-20 07:51:59 +00:00
using osuTK.Graphics;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Framework.Allocation;
using osu.Framework.Configuration;
2018-04-13 09:19:50 +00:00
using osu.Framework.Input.Bindings;
using osu.Game.Rulesets.Judgements;
using osu.Game.Rulesets.Mania.UI.Components;
2018-04-13 09:19:50 +00:00
using osu.Game.Rulesets.UI.Scrolling;
2018-11-13 05:13:29 +00:00
using OpenTK;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Rulesets.Mania.UI
{
2018-11-06 06:46:36 +00:00
public class Column : ScrollingPlayfield, IKeyBindingHandler<ManiaAction>, IHasAccentColour
2018-04-13 09:19:50 +00:00
{
private const float column_width = 45;
private const float special_column_width = 70;
2018-11-12 09:24:18 +00:00
/// <summary>
/// The index of this column as part of the whole playfield.
/// </summary>
public readonly int Index;
public readonly Bindable<ManiaAction> Action = new Bindable<ManiaAction>();
2018-04-13 09:19:50 +00:00
private readonly ColumnBackground background;
private readonly ColumnKeyArea keyArea;
private readonly ColumnHitObjectArea hitObjectArea;
2018-04-13 09:19:50 +00:00
internal readonly Container TopLevelContainer;
private readonly Container explosionContainer;
2018-11-12 09:24:18 +00:00
public Column(int index)
2018-04-13 09:19:50 +00:00
{
2018-11-12 09:24:18 +00:00
Index = index;
2018-04-13 09:19:50 +00:00
RelativeSizeAxes = Axes.Y;
Width = column_width;
2018-06-07 02:19:36 +00:00
Masking = true;
CornerRadius = 5;
background = new ColumnBackground { RelativeSizeAxes = Axes.Both };
2018-06-07 12:13:29 +00:00
2018-06-11 05:36:19 +00:00
Container hitTargetContainer;
2018-06-07 12:13:29 +00:00
InternalChildren = new[]
2018-04-13 09:19:50 +00:00
{
2018-06-07 12:13:29 +00:00
// For input purposes, the background is added at the highest depth, but is then proxied back below all other elements
background.CreateProxy(),
2018-06-11 05:36:19 +00:00
hitTargetContainer = new Container
2018-04-13 09:19:50 +00:00
{
Name = "Hit target + hit objects",
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
2018-09-21 05:35:50 +00:00
hitObjectArea = new ColumnHitObjectArea(HitObjectContainer)
{
RelativeSizeAxes = Axes.Both,
},
2018-04-13 09:19:50 +00:00
explosionContainer = new Container
{
Name = "Hit explosions",
RelativeSizeAxes = Axes.Both
}
}
},
keyArea = new ColumnKeyArea
2018-04-13 09:19:50 +00:00
{
RelativeSizeAxes = Axes.X,
Height = ManiaStage.HIT_TARGET_POSITION,
},
2018-06-07 12:13:29 +00:00
background,
2018-04-13 09:19:50 +00:00
TopLevelContainer = new Container { RelativeSizeAxes = Axes.Both }
};
TopLevelContainer.Add(explosionContainer.CreateProxy());
2018-06-11 05:36:19 +00:00
Direction.BindValueChanged(d =>
{
hitTargetContainer.Padding = new MarginPadding
{
Top = d == ScrollingDirection.Up ? ManiaStage.HIT_TARGET_POSITION : 0,
Bottom = d == ScrollingDirection.Down ? ManiaStage.HIT_TARGET_POSITION : 0,
};
2018-06-11 07:10:27 +00:00
keyArea.Anchor = keyArea.Origin= d == ScrollingDirection.Up ? Anchor.TopLeft : Anchor.BottomLeft;
2018-06-11 05:36:19 +00:00
}, true);
2018-04-13 09:19:50 +00:00
}
public override Axes RelativeSizeAxes => Axes.Y;
private bool isSpecial;
public bool IsSpecial
{
get { return isSpecial; }
set
{
if (isSpecial == value)
return;
isSpecial = value;
Width = isSpecial ? special_column_width : column_width;
}
}
private Color4 accentColour;
public Color4 AccentColour
{
get { return accentColour; }
set
{
if (accentColour == value)
return;
accentColour = value;
background.AccentColour = value;
keyArea.AccentColour = value;
hitObjectArea.AccentColour = value;
2018-04-13 09:19:50 +00:00
}
}
2018-07-11 08:07:14 +00:00
protected override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent)
{
2018-07-11 08:07:14 +00:00
var dependencies = new DependencyContainer(base.CreateChildDependencies(parent));
dependencies.CacheAs<IBindable<ManiaAction>>(Action);
return dependencies;
}
2018-04-13 09:19:50 +00:00
/// <summary>
/// Adds a DrawableHitObject to this Playfield.
/// </summary>
/// <param name="hitObject">The DrawableHitObject to add.</param>
public override void Add(DrawableHitObject hitObject)
{
hitObject.AccentColour = AccentColour;
hitObject.OnNewResult += OnNewResult;
2018-04-13 09:19:50 +00:00
HitObjectContainer.Add(hitObject);
2018-04-13 09:19:50 +00:00
}
internal void OnNewResult(DrawableHitObject judgedObject, JudgementResult result)
2018-04-13 09:19:50 +00:00
{
if (!result.IsHit || !judgedObject.DisplayResult || !DisplayJudgements)
2018-04-13 09:19:50 +00:00
return;
explosionContainer.Add(new HitExplosion(judgedObject)
{
2018-11-06 06:46:36 +00:00
Anchor = Direction.Value == ScrollingDirection.Up ? Anchor.TopCentre : Anchor.BottomCentre
});
2018-04-13 09:19:50 +00:00
}
public bool OnPressed(ManiaAction action)
{
if (action != Action)
return false;
var nextObject =
HitObjectContainer.AliveObjects.FirstOrDefault(h => h.HitObject.StartTime > Time.Current) ??
// fallback to non-alive objects to find next off-screen object
HitObjectContainer.Objects.FirstOrDefault(h => h.HitObject.StartTime > Time.Current) ??
HitObjectContainer.Objects.LastOrDefault();
nextObject?.PlaySamples();
2018-04-13 09:19:50 +00:00
return true;
}
public bool OnReleased(ManiaAction action) => false;
2018-11-13 05:13:29 +00:00
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos)
// This probably shouldn't exist as is, but the columns in the stage are separated by a 1px border
=> DrawRectangle.Inflate(new Vector2(1, 0)).Contains(ToLocalSpace(screenSpacePos));
2018-04-13 09:19:50 +00:00
}
}