2019-01-24 16:43:03 +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.
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2019-10-24 18:02:59 +08:00
|
|
|
using System;
|
2019-10-31 15:25:26 +08:00
|
|
|
using System.Collections.Generic;
|
2019-11-12 17:35:28 +08:00
|
|
|
using System.Linq;
|
2019-11-13 16:36:46 +08:00
|
|
|
using Humanizer;
|
2019-10-31 15:25:26 +08:00
|
|
|
using osu.Framework.Allocation;
|
2018-10-29 13:07:06 +08:00
|
|
|
using osu.Framework.Graphics;
|
|
|
|
using osu.Framework.Graphics.Containers;
|
2019-11-12 17:35:28 +08:00
|
|
|
using osu.Framework.Graphics.Cursor;
|
|
|
|
using osu.Framework.Graphics.UserInterface;
|
2019-10-31 16:25:30 +08:00
|
|
|
using osu.Framework.Input;
|
2019-11-05 12:26:44 +08:00
|
|
|
using osu.Framework.Input.Bindings;
|
2019-10-31 15:23:54 +08:00
|
|
|
using osu.Framework.Input.Events;
|
2019-11-12 17:35:28 +08:00
|
|
|
using osu.Game.Graphics.UserInterface;
|
2019-12-06 16:03:54 +08:00
|
|
|
using osu.Game.Rulesets.Objects;
|
2018-10-29 13:07:06 +08:00
|
|
|
using osu.Game.Rulesets.Osu.Objects;
|
2019-10-31 15:25:26 +08:00
|
|
|
using osu.Game.Screens.Edit.Compose;
|
2019-10-24 18:02:59 +08:00
|
|
|
using osuTK;
|
2019-11-13 16:28:18 +08:00
|
|
|
using osuTK.Input;
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2018-11-07 15:08:56 +08:00
|
|
|
namespace osu.Game.Rulesets.Osu.Edit.Blueprints.Sliders.Components
|
2018-10-29 13:07:06 +08:00
|
|
|
{
|
2019-11-12 17:35:28 +08:00
|
|
|
public class PathControlPointVisualiser : CompositeDrawable, IKeyBindingHandler<PlatformAction>, IHasContextMenu
|
2018-10-29 13:07:06 +08:00
|
|
|
{
|
2019-10-31 16:25:30 +08:00
|
|
|
internal readonly Container<PathControlPointPiece> Pieces;
|
2019-10-31 15:51:58 +08:00
|
|
|
private readonly Slider slider;
|
2019-11-03 18:59:37 +08:00
|
|
|
private readonly bool allowSelection;
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2019-10-31 16:25:30 +08:00
|
|
|
private InputManager inputManager;
|
|
|
|
|
2019-10-31 15:25:26 +08:00
|
|
|
[Resolved(CanBeNull = true)]
|
|
|
|
private IPlacementHandler placementHandler { get; set; }
|
|
|
|
|
2019-11-03 18:59:37 +08:00
|
|
|
public PathControlPointVisualiser(Slider slider, bool allowSelection)
|
2018-10-29 13:07:06 +08:00
|
|
|
{
|
|
|
|
this.slider = slider;
|
2019-11-03 18:59:37 +08:00
|
|
|
this.allowSelection = allowSelection;
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2019-10-31 15:23:54 +08:00
|
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
|
2019-10-31 15:51:58 +08:00
|
|
|
InternalChild = Pieces = new Container<PathControlPointPiece> { RelativeSizeAxes = Axes.Both };
|
2018-11-09 12:58:46 +08:00
|
|
|
}
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2019-10-31 16:25:30 +08:00
|
|
|
protected override void LoadComplete()
|
|
|
|
{
|
|
|
|
base.LoadComplete();
|
|
|
|
|
|
|
|
inputManager = GetContainingInputManager();
|
|
|
|
}
|
|
|
|
|
2019-09-27 17:00:24 +08:00
|
|
|
protected override void Update()
|
2018-11-09 12:58:46 +08:00
|
|
|
{
|
2019-09-27 17:00:24 +08:00
|
|
|
base.Update();
|
2018-10-29 13:07:06 +08:00
|
|
|
|
2019-12-05 18:53:31 +08:00
|
|
|
while (slider.Path.ControlPoints.Count > Pieces.Count)
|
2019-10-31 16:13:10 +08:00
|
|
|
{
|
2019-12-06 11:31:22 +08:00
|
|
|
var piece = new PathControlPointPiece(slider, Pieces.Count);
|
2019-11-03 18:59:37 +08:00
|
|
|
|
|
|
|
if (allowSelection)
|
|
|
|
piece.RequestSelection = selectPiece;
|
|
|
|
|
|
|
|
Pieces.Add(piece);
|
2019-10-31 16:13:10 +08:00
|
|
|
}
|
|
|
|
|
2019-12-05 18:53:31 +08:00
|
|
|
while (slider.Path.ControlPoints.Count < Pieces.Count)
|
2019-10-31 15:51:58 +08:00
|
|
|
Pieces.Remove(Pieces[Pieces.Count - 1]);
|
2018-10-29 13:07:06 +08:00
|
|
|
}
|
2019-10-31 15:23:54 +08:00
|
|
|
|
2019-10-31 16:13:10 +08:00
|
|
|
protected override bool OnClick(ClickEvent e)
|
2019-10-31 15:23:54 +08:00
|
|
|
{
|
2019-10-31 15:51:58 +08:00
|
|
|
foreach (var piece in Pieces)
|
2019-10-31 16:13:10 +08:00
|
|
|
piece.IsSelected.Value = false;
|
|
|
|
return false;
|
|
|
|
}
|
2019-10-31 15:23:54 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
public bool OnPressed(PlatformAction action)
|
2019-10-31 16:13:10 +08:00
|
|
|
{
|
2019-11-12 17:35:28 +08:00
|
|
|
switch (action.ActionMethod)
|
|
|
|
{
|
|
|
|
case PlatformActionMethod.Delete:
|
|
|
|
return deleteSelected();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool OnReleased(PlatformAction action) => action.ActionMethod == PlatformActionMethod.Delete;
|
|
|
|
|
2019-11-13 16:28:18 +08:00
|
|
|
private void selectPiece(int index, MouseButtonEvent e)
|
2019-10-31 16:13:10 +08:00
|
|
|
{
|
2019-11-13 16:28:18 +08:00
|
|
|
if (e.Button == MouseButton.Left && inputManager.CurrentState.Keyboard.ControlPressed)
|
2019-11-03 17:41:29 +08:00
|
|
|
Pieces[index].IsSelected.Toggle();
|
2019-10-31 16:25:30 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
foreach (var piece in Pieces)
|
|
|
|
piece.IsSelected.Value = piece.Index == index;
|
|
|
|
}
|
2019-10-31 15:23:54 +08:00
|
|
|
}
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
private bool deleteSelected()
|
2019-10-31 15:25:26 +08:00
|
|
|
{
|
2019-12-06 16:03:54 +08:00
|
|
|
List<PathControlPoint> toRemove = Pieces.Where(p => p.IsSelected.Value).Select(p => p.Index).Select(i => slider.Path.ControlPoints[i]).ToList();
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
// Ensure that there are any points to be deleted
|
2019-12-06 16:03:54 +08:00
|
|
|
if (toRemove.Count == 0)
|
2019-11-12 17:35:28 +08:00
|
|
|
return false;
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-12-06 16:03:54 +08:00
|
|
|
foreach (var c in toRemove)
|
|
|
|
slider.Path.ControlPoints.Remove(c);
|
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
// If there are 0 remaining control points, treat the slider as being deleted
|
2019-12-05 18:53:31 +08:00
|
|
|
if (slider.Path.ControlPoints.Count == 0)
|
2019-11-12 17:35:28 +08:00
|
|
|
{
|
|
|
|
placementHandler?.Delete(slider);
|
|
|
|
return true;
|
|
|
|
}
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
// Make control points relative
|
2019-12-05 18:53:31 +08:00
|
|
|
Vector2 first = slider.Path.ControlPoints[0].Position.Value;
|
|
|
|
for (int i = 0; i < slider.Path.ControlPoints.Count; i++)
|
|
|
|
slider.Path.ControlPoints[i].Position.Value = slider.Path.ControlPoints[i].Position.Value - first;
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
// The slider's position defines the position of the first control point, and all further control points are relative to that point
|
2019-11-21 23:42:46 +08:00
|
|
|
slider.Position += first;
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
// Since pieces are re-used, they will not point to the deleted control points while remaining selected
|
|
|
|
foreach (var piece in Pieces)
|
|
|
|
piece.IsSelected.Value = false;
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
return true;
|
|
|
|
}
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
public MenuItem[] ContextMenuItems
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2019-11-13 16:38:34 +08:00
|
|
|
if (!Pieces.Any(p => p.IsHovered))
|
|
|
|
return null;
|
2019-10-31 16:58:33 +08:00
|
|
|
|
2019-11-13 16:36:46 +08:00
|
|
|
int selectedPoints = Pieces.Count(p => p.IsSelected.Value);
|
2019-10-31 16:58:33 +08:00
|
|
|
|
2019-11-13 16:36:46 +08:00
|
|
|
if (selectedPoints == 0)
|
2019-11-13 15:56:48 +08:00
|
|
|
return null;
|
2019-10-31 15:25:26 +08:00
|
|
|
|
2019-11-12 17:35:28 +08:00
|
|
|
return new MenuItem[]
|
|
|
|
{
|
2019-11-13 16:36:46 +08:00
|
|
|
new OsuMenuItem($"Delete {"control point".ToQuantity(selectedPoints)}", MenuItemType.Destructive, () => deleteSelected())
|
2019-11-12 17:35:28 +08:00
|
|
|
};
|
2019-10-31 15:25:26 +08:00
|
|
|
}
|
|
|
|
}
|
2018-10-29 13:07:06 +08:00
|
|
|
}
|
|
|
|
}
|