1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-23 00:07:24 +08:00
osu-lazer/osu.Game.Rulesets.Mania/UI/ManiaPlayfield.cs

228 lines
8.6 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;
using osu.Framework.Graphics.Sprites;
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-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Mania.Judgements;
2017-05-03 11:37:47 +08:00
using osu.Framework.Graphics.Containers;
using System;
using osu.Framework.Graphics.Primitives;
using osu.Game.Graphics;
using osu.Framework.Allocation;
using OpenTK.Input;
using System.Linq;
using System.Collections.Generic;
2017-05-10 13:56:39 +08:00
using osu.Framework.Extensions.IEnumerableExtensions;
using osu.Game.Rulesets.Objects.Drawables;
using osu.Game.Rulesets.Mania.Timing;
using osu.Framework.Input;
2017-04-18 15:05:58 +08:00
namespace osu.Game.Rulesets.Mania.UI
{
2017-05-03 11:44:19 +08:00
public class ManiaPlayfield : Playfield<ManiaHitObject, ManiaJudgement>
{
public const float HIT_TARGET_POSITION = 50;
2017-05-11 18:57:24 +08:00
private const float time_span_default = 20000;
private const float time_span_min = 10;
2017-05-11 18:57:24 +08:00
private const float time_span_max = 50000;
private const float time_span_step = 1000;
/// <summary>
/// Default column keys, expanding outwards from the middle as more column are added.
/// E.g. 2 columns use FJ, 4 columns use DFJK, 6 use SDFJKL, etc...
/// </summary>
2017-05-03 19:11:24 +08:00
private static readonly Key[] default_keys = { Key.A, Key.S, Key.D, Key.F, Key.J, Key.K, Key.L, Key.Semicolon };
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;
}
}
2017-05-03 11:37:47 +08:00
public readonly FlowContainer<Column> Columns;
private readonly TimeRelativeContainer barlineContainer;
private List<Color4> normalColumnColours = new List<Color4>();
private Color4 specialColumnColour;
private readonly int columnCount;
public ManiaPlayfield(int columnCount, IEnumerable<TimingSection> timingSections)
{
this.columnCount = columnCount;
if (columnCount <= 0)
2017-05-03 11:58:46 +08:00
throw new ArgumentException("Can't have zero or fewer columns.");
2017-05-03 11:37:47 +08:00
Children = new Drawable[]
{
new Container
{
2017-05-11 13:26:00 +08:00
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
RelativeSizeAxes = Axes.Y,
2017-05-03 11:37:47 +08:00
AutoSizeAxes = Axes.X,
2017-05-11 19:04:28 +08:00
Masking = true,
2017-05-03 11:37:47 +08:00
Children = new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = Color4.Black
},
Columns = new FillFlowContainer<Column>
{
Name = "Columns",
2017-05-03 11:37:47 +08:00
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Direction = FillDirection.Horizontal,
Padding = new MarginPadding { Left = 1, Right = 1 },
Spacing = new Vector2(1, 0)
},
barlineContainer = new TimeRelativeContainer(timingSections)
{
2017-05-11 21:01:37 +08:00
Name = "Bar lines",
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding { Bottom = HIT_TARGET_POSITION }
2017-05-03 11:37:47 +08:00
}
}
}
};
for (int i = 0; i < columnCount; i++)
Columns.Add(new Column(timingSections));
TimeSpan = time_span_default;
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
for (int i = 0; i < columnCount; i++)
{
Column column = Columns.Children.ElementAt(i);
column.IsSpecial = isSpecialColumn(i);
if (!column.IsSpecial)
continue;
column.Key = Key.Space;
column.AccentColour = specialColumnColour;
}
var nonSpecialColumns = Columns.Children.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
// We'll set the keys for non-special columns in another separate loop because it's not mirrored like the above colours
// Todo: This needs to go when we get to bindings and use Button1, ..., ButtonN instead
for (int i = 0; i < nonSpecialColumns.Count; i++)
2017-05-03 11:37:47 +08:00
{
Column column = nonSpecialColumns[i];
2017-05-03 11:37:47 +08:00
int keyOffset = default_keys.Length / 2 - nonSpecialColumns.Count / 2 + i;
if (keyOffset >= 0 && keyOffset < default_keys.Length)
column.Key = default_keys[keyOffset];
2017-05-03 18:42:20 +08:00
else
// There is no default key defined for this column. Let's set this to Unknown for now
// however note that this will be gone after bindings are in place
column.Key = Key.Unknown;
2017-05-03 11:37:47 +08:00
}
}
2017-05-03 11:37:47 +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<ManiaHitObject, ManiaJudgement> h) => Columns.Children.ElementAt(h.HitObject.Column).Add(h);
protected override bool OnKeyDown(InputState state, KeyDownEventArgs args)
{
if (args.Repeat)
return false;
if (state.Keyboard.ControlPressed)
{
switch (args.Key)
{
case Key.Minus:
TimeSpan += time_span_step;
break;
case Key.Plus:
TimeSpan -= time_span_step;
break;
}
}
return false;
}
private double timeSpan;
/// <summary>
/// The amount of time which the length of the playfield spans.
/// </summary>
public double TimeSpan
{
get { return timeSpan; }
set
{
if (timeSpan == value)
return;
timeSpan = value;
barlineContainer.TimeSpan = value;
Columns.Children.ForEach(c => c.TimingSectionContainer.TimeSpan = value);
}
}
}
2017-05-03 12:03:46 +08:00
}