1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-15 21:03:08 +08:00
osu-lazer/osu.Game.Rulesets.Mania/UI/ManiaPlayfield.cs

239 lines
9.3 KiB
C#
Raw Normal View History

// Copyright (c) 2007-2017 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using osu.Framework.Graphics;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Mania.Objects;
using osu.Game.Rulesets.UI;
using OpenTK;
using OpenTK.Graphics;
2017-05-03 11:37:47 +08:00
using osu.Framework.Graphics.Containers;
using System;
using osu.Game.Graphics;
using osu.Framework.Allocation;
using System.Linq;
using System.Collections.Generic;
using osu.Framework.Configuration;
2017-05-10 13:56:39 +08:00
using osu.Game.Rulesets.Objects.Drawables;
2017-05-29 13:44:42 +08:00
using osu.Game.Rulesets.Mania.Objects.Drawables;
using osu.Framework.Graphics.Shapes;
using osu.Game.Rulesets.Judgements;
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Mania.UI
{
public class ManiaPlayfield : ScrollingPlayfield
{
public const float HIT_TARGET_POSITION = 50;
private SpecialColumnPosition specialColumnPosition;
/// <summary>
/// The style to use for the special column.
/// </summary>
public SpecialColumnPosition SpecialColumnPosition
{
get { return specialColumnPosition; }
set
{
if (IsLoaded)
throw new InvalidOperationException($"Setting {nameof(SpecialColumnPosition)} after the playfield is loaded requires re-creating the playfield.");
specialColumnPosition = value;
}
}
/// <summary>
/// Whether this playfield should be inverted. This flips everything inside the playfield.
/// </summary>
public readonly Bindable<bool> Inverted = new Bindable<bool>(true);
private readonly FlowContainer<Column> columns;
public IEnumerable<Column> Columns => columns.Children;
2017-08-07 16:45:45 +08:00
protected override Container<Drawable> Content => content;
private readonly Container<Drawable> content;
private List<Color4> normalColumnColours = new List<Color4>();
private Color4 specialColumnColour;
private readonly Container<DrawableManiaJudgement> judgements;
private readonly int columnCount;
public ManiaPlayfield(int columnCount)
: base(Axes.Y)
{
this.columnCount = columnCount;
if (columnCount <= 0)
2017-05-03 11:58:46 +08:00
throw new ArgumentException("Can't have zero or fewer columns.");
Inverted.Value = true;
2017-09-11 12:44:39 +08:00
Container topLevelContainer;
InternalChildren = new Drawable[]
2017-05-03 11:37:47 +08:00
{
new Container
{
Name = "Playfield elements",
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
2017-05-03 11:37:47 +08:00
Children = new Drawable[]
{
2017-05-29 13:44:42 +08:00
new Container
2017-05-03 11:37:47 +08:00
{
Name = "Columns mask",
2017-05-03 11:37:47 +08:00
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
2017-05-29 13:44:42 +08:00
Masking = true,
Children = new Drawable[]
{
new Box
{
Name = "Background",
2017-05-29 13:44:42 +08:00
RelativeSizeAxes = Axes.Both,
Colour = Color4.Black
},
columns = new FillFlowContainer<Column>
{
Name = "Columns",
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Direction = FillDirection.Horizontal,
Padding = new MarginPadding { Left = 1, Right = 1 },
Spacing = new Vector2(1, 0)
},
2017-05-29 13:44:42 +08:00
}
},
new Container
{
Name = "Barlines mask",
2017-05-29 13:44:42 +08:00
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.Y,
Width = 1366, // Bar lines should only be masked on the vertical axis
BypassAutoSizeAxes = Axes.Both,
Masking = true,
Child = content = new Container
{
Name = "Bar lines",
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.Y,
Padding = new MarginPadding { Top = HIT_TARGET_POSITION }
}
},
judgements = new Container<DrawableManiaJudgement>
{
Anchor = Anchor.TopCentre,
Origin = Anchor.Centre,
AutoSizeAxes = Axes.Both,
Y = HIT_TARGET_POSITION + 150,
BypassAutoSizeAxes = Axes.Both
},
topLevelContainer = new Container { RelativeSizeAxes = Axes.Both }
2017-05-03 11:37:47 +08:00
}
}
};
2017-08-23 12:42:11 +08:00
var currentAction = ManiaAction.Key1;
for (int i = 0; i < columnCount; i++)
{
var c = new Column();
c.VisibleTimeRange.BindTo(VisibleTimeRange);
2017-08-23 12:42:11 +08:00
c.IsSpecial = isSpecialColumn(i);
c.Action = c.IsSpecial ? ManiaAction.Special : currentAction++;
2017-09-11 12:44:39 +08:00
topLevelContainer.Add(c.TopLevelContainer.CreateProxy());
columns.Add(c);
AddNested(c);
}
Inverted.ValueChanged += invertedChanged;
Inverted.TriggerChange();
}
private void invertedChanged(bool newValue)
{
Scale = new Vector2(1, newValue ? -1 : 1);
judgements.Scale = Scale;
2017-05-03 11:37:47 +08:00
}
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
normalColumnColours = new List<Color4>
2017-05-03 11:37:47 +08:00
{
colours.RedDark,
colours.GreenDark
2017-05-03 11:37:47 +08:00
};
specialColumnColour = colours.BlueDark;
// Set the special column + colour + key
2017-08-23 12:42:11 +08:00
foreach (var column in Columns)
{
if (!column.IsSpecial)
continue;
column.AccentColour = specialColumnColour;
}
var nonSpecialColumns = Columns.Where(c => !c.IsSpecial).ToList();
2017-05-03 11:37:47 +08:00
// We'll set the colours of the non-special columns in a separate loop, because the non-special
// column colours are mirrored across their centre and special styles mess with this
for (int i = 0; i < Math.Ceiling(nonSpecialColumns.Count / 2f); i++)
{
Color4 colour = normalColumnColours[i % normalColumnColours.Count];
nonSpecialColumns[i].AccentColour = colour;
nonSpecialColumns[nonSpecialColumns.Count - 1 - i].AccentColour = colour;
}
}
2017-05-03 11:37:47 +08:00
public override void OnJudgement(DrawableHitObject judgedObject, Judgement judgement)
{
var maniaObject = (ManiaHitObject)judgedObject.HitObject;
columns[maniaObject.Column].OnJudgement(judgedObject, judgement);
judgements.Clear();
judgements.Add(new DrawableManiaJudgement(judgement)
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
});
}
2017-09-11 11:57:10 +08:00
/// <summary>
/// Whether the column index is a special column for this playfield.
/// </summary>
/// <param name="column">The 0-based column index.</param>
/// <returns>Whether the column is a special column.</returns>
private bool isSpecialColumn(int column)
{
2017-05-04 13:46:10 +08:00
switch (SpecialColumnPosition)
2017-05-03 11:37:47 +08:00
{
default:
2017-05-04 13:46:10 +08:00
case SpecialColumnPosition.Normal:
return columnCount % 2 == 1 && column == columnCount / 2;
2017-05-04 13:46:10 +08:00
case SpecialColumnPosition.Left:
return column == 0;
2017-05-04 13:46:10 +08:00
case SpecialColumnPosition.Right:
return column == columnCount - 1;
2017-05-03 11:37:47 +08:00
}
}
2017-05-10 13:56:39 +08:00
public override void Add(DrawableHitObject h) => Columns.ElementAt(((ManiaHitObject)h.HitObject).Column).Add(h);
2017-08-07 16:45:45 +08:00
public void Add(DrawableBarLine barline) => HitObjects.Add(barline);
2017-05-16 18:14:27 +08:00
2017-05-29 13:44:42 +08:00
protected override void Update()
{
// Due to masking differences, it is not possible to get the width of the columns container automatically
// While masking on effectively only the Y-axis, so we need to set the width of the bar line container manually
2017-08-07 16:45:45 +08:00
content.Width = columns.Width;
2017-05-29 13:44:42 +08:00
}
}
2017-05-03 12:03:46 +08:00
}