2023-12-28 21:36:30 +00: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.
|
|
|
|
|
|
2024-01-01 14:46:07 +00:00
|
|
|
|
using System.Linq;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
|
using osu.Framework.Graphics;
|
2024-01-01 14:46:07 +00:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Graphics.Shapes;
|
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
2023-12-28 22:10:06 +00:00
|
|
|
|
using osu.Framework.Input.Bindings;
|
|
|
|
|
using osu.Framework.Input.Events;
|
2024-06-05 16:12:02 +00:00
|
|
|
|
using osu.Game.Graphics.Containers;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
2023-12-28 22:10:06 +00:00
|
|
|
|
using osu.Game.Input.Bindings;
|
|
|
|
|
using osu.Game.Rulesets.Edit;
|
|
|
|
|
using osu.Game.Rulesets.Osu.UI;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
using osu.Game.Screens.Edit;
|
2024-01-01 14:46:07 +00:00
|
|
|
|
using osu.Game.Screens.Edit.Components.RadioButtons;
|
2023-12-29 23:43:41 +00:00
|
|
|
|
using osuTK;
|
2024-01-01 14:46:07 +00:00
|
|
|
|
using osuTK.Graphics;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
|
2023-12-28 22:10:06 +00:00
|
|
|
|
namespace osu.Game.Rulesets.Osu.Edit
|
2023-12-28 21:36:30 +00:00
|
|
|
|
{
|
2023-12-28 22:10:06 +00:00
|
|
|
|
public partial class OsuGridToolboxGroup : EditorToolboxGroup, IKeyBindingHandler<GlobalAction>
|
2023-12-28 21:36:30 +00:00
|
|
|
|
{
|
|
|
|
|
[Resolved]
|
|
|
|
|
private EditorBeatmap editorBeatmap { get; set; } = null!;
|
|
|
|
|
|
2024-06-05 16:12:02 +00:00
|
|
|
|
[Resolved]
|
|
|
|
|
private IExpandingContainer? expandingContainer { get; set; }
|
|
|
|
|
|
2023-12-29 23:43:41 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// X position of the grid's origin.
|
|
|
|
|
/// </summary>
|
2023-12-28 22:10:06 +00:00
|
|
|
|
public BindableFloat StartPositionX { get; } = new BindableFloat(OsuPlayfield.BASE_SIZE.X / 2)
|
2023-12-28 21:36:30 +00:00
|
|
|
|
{
|
|
|
|
|
MinValue = 0f,
|
2023-12-28 22:10:06 +00:00
|
|
|
|
MaxValue = OsuPlayfield.BASE_SIZE.X,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
Precision = 1f
|
|
|
|
|
};
|
|
|
|
|
|
2023-12-29 23:43:41 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Y position of the grid's origin.
|
|
|
|
|
/// </summary>
|
2023-12-28 22:10:06 +00:00
|
|
|
|
public BindableFloat StartPositionY { get; } = new BindableFloat(OsuPlayfield.BASE_SIZE.Y / 2)
|
2023-12-28 21:36:30 +00:00
|
|
|
|
{
|
|
|
|
|
MinValue = 0f,
|
2023-12-28 22:10:06 +00:00
|
|
|
|
MaxValue = OsuPlayfield.BASE_SIZE.Y,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
Precision = 1f
|
|
|
|
|
};
|
|
|
|
|
|
2023-12-29 23:43:41 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The spacing between grid lines.
|
|
|
|
|
/// </summary>
|
2023-12-28 21:36:30 +00:00
|
|
|
|
public BindableFloat Spacing { get; } = new BindableFloat(4f)
|
|
|
|
|
{
|
|
|
|
|
MinValue = 4f,
|
|
|
|
|
MaxValue = 128f,
|
|
|
|
|
Precision = 1f
|
|
|
|
|
};
|
|
|
|
|
|
2023-12-29 23:43:41 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Rotation of the grid lines in degrees.
|
|
|
|
|
/// </summary>
|
2023-12-28 21:36:30 +00:00
|
|
|
|
public BindableFloat GridLinesRotation { get; } = new BindableFloat(0f)
|
|
|
|
|
{
|
2024-06-19 18:46:55 +00:00
|
|
|
|
MinValue = -180f,
|
|
|
|
|
MaxValue = 180f,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
Precision = 1f
|
|
|
|
|
};
|
|
|
|
|
|
2023-12-29 23:43:41 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Read-only bindable representing the grid's origin.
|
|
|
|
|
/// Equivalent to <code>new Vector2(StartPositionX, StartPositionY)</code>
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Bindable<Vector2> StartPosition { get; } = new Bindable<Vector2>();
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Read-only bindable representing the grid's spacing in both the X and Y dimension.
|
|
|
|
|
/// Equivalent to <code>new Vector2(Spacing)</code>
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Bindable<Vector2> SpacingVector { get; } = new Bindable<Vector2>();
|
|
|
|
|
|
2024-01-01 14:46:07 +00:00
|
|
|
|
public Bindable<PositionSnapGridType> GridType { get; } = new Bindable<PositionSnapGridType>();
|
|
|
|
|
|
2023-12-28 21:36:30 +00:00
|
|
|
|
private ExpandableSlider<float> startPositionXSlider = null!;
|
|
|
|
|
private ExpandableSlider<float> startPositionYSlider = null!;
|
|
|
|
|
private ExpandableSlider<float> spacingSlider = null!;
|
|
|
|
|
private ExpandableSlider<float> gridLinesRotationSlider = null!;
|
2024-01-01 14:46:07 +00:00
|
|
|
|
private EditorRadioButtonCollection gridTypeButtons = null!;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
|
2023-12-28 22:10:06 +00:00
|
|
|
|
public OsuGridToolboxGroup()
|
|
|
|
|
: base("grid")
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-31 20:57:11 +00:00
|
|
|
|
private const float max_automatic_spacing = 64;
|
|
|
|
|
|
2023-12-28 21:36:30 +00:00
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load()
|
|
|
|
|
{
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
startPositionXSlider = new ExpandableSlider<float>
|
|
|
|
|
{
|
2023-12-31 15:41:22 +00:00
|
|
|
|
Current = StartPositionX,
|
|
|
|
|
KeyboardStep = 1,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
},
|
|
|
|
|
startPositionYSlider = new ExpandableSlider<float>
|
|
|
|
|
{
|
2023-12-31 15:41:22 +00:00
|
|
|
|
Current = StartPositionY,
|
|
|
|
|
KeyboardStep = 1,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
},
|
|
|
|
|
spacingSlider = new ExpandableSlider<float>
|
|
|
|
|
{
|
2023-12-31 15:41:22 +00:00
|
|
|
|
Current = Spacing,
|
|
|
|
|
KeyboardStep = 1,
|
2023-12-28 21:36:30 +00:00
|
|
|
|
},
|
|
|
|
|
gridLinesRotationSlider = new ExpandableSlider<float>
|
|
|
|
|
{
|
2023-12-31 15:41:22 +00:00
|
|
|
|
Current = GridLinesRotation,
|
|
|
|
|
KeyboardStep = 1,
|
2023-12-29 15:59:12 +00:00
|
|
|
|
},
|
2024-01-01 14:46:07 +00:00
|
|
|
|
new FillFlowContainer
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Spacing = new Vector2(0f, 10f),
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
gridTypeButtons = new EditorRadioButtonCollection
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
Items = new[]
|
|
|
|
|
{
|
|
|
|
|
new RadioButton("Square",
|
|
|
|
|
() => GridType.Value = PositionSnapGridType.Square,
|
|
|
|
|
() => new SpriteIcon { Icon = FontAwesome.Regular.Square }),
|
|
|
|
|
new RadioButton("Triangle",
|
|
|
|
|
() => GridType.Value = PositionSnapGridType.Triangle,
|
|
|
|
|
() => new OutlineTriangle(true, 20)),
|
|
|
|
|
new RadioButton("Circle",
|
|
|
|
|
() => GridType.Value = PositionSnapGridType.Circle,
|
|
|
|
|
() => new SpriteIcon { Icon = FontAwesome.Regular.Circle }),
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
},
|
2023-12-28 21:36:30 +00:00
|
|
|
|
};
|
2023-12-28 22:10:06 +00:00
|
|
|
|
|
2023-12-30 13:32:20 +00:00
|
|
|
|
Spacing.Value = editorBeatmap.BeatmapInfo.GridSize;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
|
|
|
|
|
2024-01-01 14:46:07 +00:00
|
|
|
|
gridTypeButtons.Items.First().Select();
|
|
|
|
|
|
2023-12-28 21:36:30 +00:00
|
|
|
|
StartPositionX.BindValueChanged(x =>
|
|
|
|
|
{
|
|
|
|
|
startPositionXSlider.ContractedLabelText = $"X: {x.NewValue:N0}";
|
|
|
|
|
startPositionXSlider.ExpandedLabelText = $"X Offset: {x.NewValue:N0}";
|
2023-12-29 23:43:41 +00:00
|
|
|
|
StartPosition.Value = new Vector2(x.NewValue, StartPosition.Value.Y);
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}, true);
|
|
|
|
|
|
|
|
|
|
StartPositionY.BindValueChanged(y =>
|
|
|
|
|
{
|
|
|
|
|
startPositionYSlider.ContractedLabelText = $"Y: {y.NewValue:N0}";
|
|
|
|
|
startPositionYSlider.ExpandedLabelText = $"Y Offset: {y.NewValue:N0}";
|
2023-12-29 23:43:41 +00:00
|
|
|
|
StartPosition.Value = new Vector2(StartPosition.Value.X, y.NewValue);
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}, true);
|
|
|
|
|
|
|
|
|
|
Spacing.BindValueChanged(spacing =>
|
|
|
|
|
{
|
|
|
|
|
spacingSlider.ContractedLabelText = $"S: {spacing.NewValue:N0}";
|
|
|
|
|
spacingSlider.ExpandedLabelText = $"Spacing: {spacing.NewValue:N0}";
|
2023-12-29 23:43:41 +00:00
|
|
|
|
SpacingVector.Value = new Vector2(spacing.NewValue);
|
2023-12-30 13:32:20 +00:00
|
|
|
|
editorBeatmap.BeatmapInfo.GridSize = (int)spacing.NewValue;
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}, true);
|
|
|
|
|
|
|
|
|
|
GridLinesRotation.BindValueChanged(rotation =>
|
|
|
|
|
{
|
2023-12-31 15:45:21 +00:00
|
|
|
|
gridLinesRotationSlider.ContractedLabelText = $"R: {rotation.NewValue:#,0.##}";
|
|
|
|
|
gridLinesRotationSlider.ExpandedLabelText = $"Rotation: {rotation.NewValue:#,0.##}";
|
|
|
|
|
}, true);
|
2024-01-01 14:46:07 +00:00
|
|
|
|
|
|
|
|
|
expandingContainer?.Expanded.BindValueChanged(v =>
|
|
|
|
|
{
|
|
|
|
|
gridTypeButtons.FadeTo(v.NewValue ? 1f : 0f, 500, Easing.OutQuint);
|
|
|
|
|
gridTypeButtons.BypassAutoSizeAxes = !v.NewValue ? Axes.Y : Axes.None;
|
|
|
|
|
}, true);
|
2024-06-19 18:46:55 +00:00
|
|
|
|
|
|
|
|
|
GridType.BindValueChanged(v =>
|
|
|
|
|
{
|
|
|
|
|
GridLinesRotation.Disabled = v.NewValue == PositionSnapGridType.Circle;
|
|
|
|
|
|
|
|
|
|
switch (v.NewValue)
|
|
|
|
|
{
|
|
|
|
|
case PositionSnapGridType.Square:
|
2024-06-19 19:10:30 +00:00
|
|
|
|
GridLinesRotation.Value = ((GridLinesRotation.Value + 405) % 90) - 45;
|
2024-06-19 18:46:55 +00:00
|
|
|
|
GridLinesRotation.MinValue = -45;
|
|
|
|
|
GridLinesRotation.MaxValue = 45;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case PositionSnapGridType.Triangle:
|
2024-06-19 19:10:30 +00:00
|
|
|
|
GridLinesRotation.Value = ((GridLinesRotation.Value + 390) % 60) - 30;
|
2024-06-19 18:46:55 +00:00
|
|
|
|
GridLinesRotation.MinValue = -30;
|
|
|
|
|
GridLinesRotation.MaxValue = 30;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}, true);
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}
|
2023-12-28 22:10:06 +00:00
|
|
|
|
|
2023-12-31 20:57:11 +00:00
|
|
|
|
private void nextGridSize()
|
2023-12-28 22:10:06 +00:00
|
|
|
|
{
|
2023-12-31 20:57:11 +00:00
|
|
|
|
Spacing.Value = Spacing.Value * 2 >= max_automatic_spacing ? Spacing.Value / 8 : Spacing.Value * 2;
|
2023-12-28 22:10:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
|
|
|
|
|
{
|
|
|
|
|
switch (e.Action)
|
|
|
|
|
{
|
|
|
|
|
case GlobalAction.EditorCycleGridDisplayMode:
|
2023-12-31 20:57:11 +00:00
|
|
|
|
nextGridSize();
|
2023-12-28 22:10:06 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
|
|
|
|
|
{
|
|
|
|
|
}
|
2024-01-01 14:46:07 +00:00
|
|
|
|
|
|
|
|
|
public partial class OutlineTriangle : BufferedContainer
|
|
|
|
|
{
|
|
|
|
|
public OutlineTriangle(bool outlineOnly, float size)
|
|
|
|
|
: base(cachedFrameBuffer: true)
|
|
|
|
|
{
|
|
|
|
|
Size = new Vector2(size);
|
|
|
|
|
|
|
|
|
|
InternalChildren = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
new EquilateralTriangle { RelativeSizeAxes = Axes.Both },
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (outlineOnly)
|
|
|
|
|
{
|
|
|
|
|
AddInternal(new EquilateralTriangle
|
|
|
|
|
{
|
|
|
|
|
Anchor = Anchor.TopCentre,
|
|
|
|
|
Origin = Anchor.Centre,
|
|
|
|
|
RelativePositionAxes = Axes.Y,
|
|
|
|
|
Y = 0.48f,
|
|
|
|
|
Colour = Color4.Black,
|
|
|
|
|
Size = new Vector2(size - 7),
|
|
|
|
|
Blending = BlendingParameters.None,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Blending = BlendingParameters.Additive;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public enum PositionSnapGridType
|
|
|
|
|
{
|
|
|
|
|
Square,
|
|
|
|
|
Triangle,
|
|
|
|
|
Circle,
|
2023-12-29 15:59:12 +00:00
|
|
|
|
}
|
2023-12-28 21:36:30 +00:00
|
|
|
|
}
|