// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. using System; using System.Linq; using osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Graphics.Pooling; using osu.Framework.Graphics.Shapes; using osu.Framework.Graphics.UserInterface; using osu.Framework.Input.Events; using osu.Framework.Localisation; using osu.Game.Beatmaps.ControlPoints; using osu.Game.Extensions; using osu.Game.Graphics; using osu.Game.Graphics.Containers; using osu.Game.Graphics.Sprites; using osu.Game.Overlays; using osu.Game.Screens.Edit.Timing.RowAttributes; using osuTK; namespace osu.Game.Screens.Edit.Timing { public partial class ControlPointTable : CompositeDrawable { public BindableList Groups { get; } = new BindableList(); private const float timing_column_width = 300; private const float row_height = 25; private const float row_horizontal_padding = 20; [BackgroundDependencyLoader] private void load(OverlayColourProvider colours) { RelativeSizeAxes = Axes.Both; InternalChildren = new Drawable[] { new Box { Colour = colours.Background4, RelativeSizeAxes = Axes.Both, }, new Box { Colour = colours.Background3, RelativeSizeAxes = Axes.Y, Width = timing_column_width + 10, }, new Container { RelativeSizeAxes = Axes.X, Height = row_height, Padding = new MarginPadding { Horizontal = row_horizontal_padding }, Children = new Drawable[] { new TableHeaderText("Time") { Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, }, new TableHeaderText("Attributes") { Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, Margin = new MarginPadding { Left = ControlPointTable.timing_column_width } }, } }, new Container { RelativeSizeAxes = Axes.Both, Padding = new MarginPadding { Top = row_height }, Child = new ControlPointRowList { RelativeSizeAxes = Axes.Both, RowData = { BindTarget = Groups, }, }, }, }; } private partial class ControlPointRowList : VirtualisedListContainer { [Resolved] private Bindable selectedGroup { get; set; } = null!; public ControlPointRowList() : base(row_height, 50) { } protected override ScrollContainer CreateScrollContainer() => new OsuScrollContainer(); protected override void LoadComplete() { base.LoadComplete(); selectedGroup.BindValueChanged(val => { // can't use `.ScrollIntoView()` here because of the list virtualisation not giving // child items valid coordinates from the start, so ballpark something similar // using estimated row height. var row = Items.FlowingChildren.SingleOrDefault(item => item.Row.Equals(val.NewValue)); if (row == null) return; float minPos = Items.GetLayoutPosition(row) * row_height; float maxPos = minPos + row_height; if (minPos < Scroll.Current) Scroll.ScrollTo(minPos); else if (maxPos > Scroll.Current + Scroll.DisplayableContent) Scroll.ScrollTo(maxPos - Scroll.DisplayableContent); }); } } public partial class DrawableControlGroup : PoolableDrawable, IHasCurrentValue { public Bindable Current { get => current.Current; set => current.Current = value; } private readonly BindableWithCurrent current = new BindableWithCurrent(); private Box background = null!; [Resolved] private OverlayColourProvider colourProvider { get; set; } = null!; [Resolved] private Bindable selectedGroup { get; set; } = null!; [Resolved] private EditorClock editorClock { get; set; } = null!; [BackgroundDependencyLoader] private void load() { RelativeSizeAxes = Axes.Both; InternalChildren = new Drawable[] { background = new Box { RelativeSizeAxes = Axes.Both, Colour = colourProvider.Background1, Alpha = 0, }, new Container { RelativeSizeAxes = Axes.Both, Padding = new MarginPadding { Horizontal = row_horizontal_padding, }, Children = new Drawable[] { new ControlGroupTiming { Group = { BindTarget = current }, }, new ControlGroupAttributes(point => point is not TimingControlPoint) { Group = { BindTarget = current }, Margin = new MarginPadding { Left = timing_column_width } } } } }; } protected override void LoadComplete() { base.LoadComplete(); selectedGroup.BindValueChanged(_ => updateState(), true); FinishTransforms(true); } protected override void PrepareForUse() { base.PrepareForUse(); updateState(); } protected override bool OnHover(HoverEvent e) { updateState(); return true; } protected override void OnHoverLost(HoverLostEvent e) { base.OnHoverLost(e); updateState(); } protected override bool OnClick(ClickEvent e) { // schedule to give time for any modified focused text box to lose focus and commit changes (e.g. BPM / time signature textboxes) before switching to new point. var currentGroup = Current.Value; Schedule(() => { selectedGroup.Value = currentGroup; editorClock.SeekSmoothlyTo(currentGroup.Time); }); return true; } private void updateState() { bool isSelected = selectedGroup.Value?.Equals(current.Value) == true; if (IsHovered || isSelected) background.FadeIn(100, Easing.OutQuint); else background.FadeOut(100, Easing.OutQuint); background.Colour = isSelected ? colourProvider.Colour3 : colourProvider.Background1; } } private partial class ControlGroupTiming : FillFlowContainer { public Bindable Group { get; } = new Bindable(); private OsuSpriteText timeText = null!; [BackgroundDependencyLoader] private void load() { Name = @"ControlGroupTiming"; RelativeSizeAxes = Axes.Y; Width = timing_column_width; Spacing = new Vector2(5); Children = new Drawable[] { timeText = new OsuSpriteText { Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold), Width = 70, Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, }, new ControlGroupAttributes(c => c is TimingControlPoint) { Anchor = Anchor.CentreLeft, Origin = Anchor.CentreLeft, Group = { BindTarget = Group }, } }; } protected override void LoadComplete() { base.LoadComplete(); Group.BindValueChanged(_ => timeText.Text = Group.Value?.Time.ToEditorFormattedString() ?? default(LocalisableString), true); } } private partial class ControlGroupAttributes : CompositeDrawable { public Bindable Group { get; } = new Bindable(); private BindableList controlPoints { get; } = new BindableList(); private readonly Func matchFunction; private FillFlowContainer fill = null!; public ControlGroupAttributes(Func matchFunction) { this.matchFunction = matchFunction; } [BackgroundDependencyLoader] private void load() { AutoSizeAxes = Axes.X; RelativeSizeAxes = Axes.Y; Name = @"ControlGroupAttributes"; InternalChild = fill = new FillFlowContainer { AutoSizeAxes = Axes.X, RelativeSizeAxes = Axes.Y, Direction = FillDirection.Horizontal, Spacing = new Vector2(2) }; } protected override void LoadComplete() { base.LoadComplete(); Group.BindValueChanged(_ => { controlPoints.UnbindBindings(); controlPoints.Clear(); if (Group.Value != null) ((IBindableList)controlPoints).BindTo(Group.Value.ControlPoints); }, true); controlPoints.BindCollectionChanged((_, _) => createChildren(), true); } private void createChildren() { fill.ChildrenEnumerable = controlPoints .Where(matchFunction) .Select(createAttribute) // arbitrary ordering to make timing points first. // probably want to explicitly define order in the future. .OrderByDescending(c => c.GetType().Name); } 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); } throw new ArgumentOutOfRangeException(nameof(controlPoint), $"Control point type {controlPoint.GetType()} is not supported"); } } } }