1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-07 20:03:13 +08:00
osu-lazer/osu.Game/Screens/Edit/Timing/ControlPointTable.cs

180 lines
6.0 KiB
C#
Raw Normal View History

2019-10-18 16:59:54 +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.
using System;
2019-10-18 16:59:54 +08:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Beatmaps.ControlPoints;
using osu.Game.Extensions;
2019-10-18 16:59:54 +08:00
using osu.Game.Graphics;
using osu.Game.Graphics.Sprites;
using osu.Game.Screens.Edit.Timing.RowAttributes;
2019-10-18 16:59:54 +08:00
using osuTK;
namespace osu.Game.Screens.Edit.Timing
{
public class ControlPointTable : EditorTable
2019-10-18 16:59:54 +08:00
{
[Resolved]
private Bindable<ControlPointGroup> selectedGroup { get; set; }
2019-10-18 16:59:54 +08:00
[Resolved]
private EditorClock clock { get; set; }
public const float TIMING_COLUMN_WIDTH = 230;
public IEnumerable<ControlPointGroup> ControlGroups
2019-10-18 16:59:54 +08:00
{
set
{
Content = null;
BackgroundFlow.Clear();
2019-10-18 16:59:54 +08:00
if (value?.Any() != true)
return;
foreach (var group in value)
2019-10-18 16:59:54 +08:00
{
BackgroundFlow.Add(new RowBackground(group)
{
Action = () =>
{
selectedGroup.Value = group;
clock.SeekSmoothlyTo(group.Time);
}
});
2019-10-18 16:59:54 +08:00
}
Columns = createHeaders();
Content = value.Select((g, i) => createContent(i, g)).ToArray().ToRectangular();
2019-10-18 16:59:54 +08:00
}
}
protected override void LoadComplete()
{
base.LoadComplete();
selectedGroup.BindValueChanged(group =>
{
foreach (var b in BackgroundFlow) b.Selected = b.Item == group.NewValue;
}, true);
}
2019-10-18 16:59:54 +08:00
private TableColumn[] createHeaders()
{
var columns = new List<TableColumn>
{
new TableColumn("Time", Anchor.CentreLeft, new Dimension(GridSizeMode.Absolute, TIMING_COLUMN_WIDTH)),
new TableColumn("Attributes", Anchor.CentreLeft),
2019-10-18 16:59:54 +08:00
};
return columns.ToArray();
}
private Drawable[] createContent(int index, ControlPointGroup group)
2019-10-18 16:59:54 +08:00
{
return new Drawable[]
2019-10-18 16:59:54 +08:00
{
new FillFlowContainer
{
RelativeSizeAxes = Axes.Y,
Width = TIMING_COLUMN_WIDTH,
Spacing = new Vector2(5),
Children = new Drawable[]
{
new OsuSpriteText
{
Text = group.Time.ToEditorFormattedString(),
Font = OsuFont.GetFont(size: TEXT_SIZE, weight: FontWeight.Bold),
Width = 70,
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
},
new ControlGroupAttributes(group, c => c is TimingControlPoint)
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
}
}
},
new ControlGroupAttributes(group, c => !(c is TimingControlPoint))
};
}
2019-10-18 16:59:54 +08:00
private class ControlGroupAttributes : CompositeDrawable
2019-10-18 16:59:54 +08:00
{
private readonly Func<ControlPoint, bool> matchFunction;
private readonly IBindableList<ControlPoint> controlPoints = new BindableList<ControlPoint>();
2019-10-18 16:59:54 +08:00
private readonly FillFlowContainer fill;
2019-10-18 16:59:54 +08:00
public ControlGroupAttributes(ControlPointGroup group, Func<ControlPoint, bool> matchFunction)
{
this.matchFunction = matchFunction;
AutoSizeAxes = Axes.X;
RelativeSizeAxes = Axes.Y;
InternalChild = fill = new FillFlowContainer
{
AutoSizeAxes = Axes.X,
RelativeSizeAxes = Axes.Y,
Direction = FillDirection.Horizontal,
Spacing = new Vector2(2)
};
controlPoints.BindTo(group.ControlPoints);
}
[BackgroundDependencyLoader]
private void load()
{
createChildren();
}
2019-10-18 16:59:54 +08:00
protected override void LoadComplete()
{
base.LoadComplete();
controlPoints.CollectionChanged += (_, __) => createChildren();
}
private void createChildren()
{
fill.ChildrenEnumerable = controlPoints
.Where(matchFunction)
.Select(createAttribute)
.Where(c => c != null)
// arbitrary ordering to make timing points first.
// probably want to explicitly define order in the future.
.OrderByDescending(c => c.GetType().Name);
2019-10-18 16:59:54 +08:00
}
private Drawable createAttribute(ControlPoint controlPoint)
{
switch (controlPoint)
{
case TimingControlPoint timing:
return new TimingRowAttribute(timing);
case DifficultyControlPoint difficulty:
return new DifficultyRowAttribute(difficulty);
case EffectControlPoint effect:
return new EffectRowAttribute(effect);
case SampleControlPoint sample:
return new SampleRowAttribute(sample);
}
return null;
}
2019-10-18 16:59:54 +08:00
}
}
}