1
0
mirror of https://github.com/ppy/osu.git synced 2024-12-17 05:22:54 +08:00
osu-lazer/osu.Game.Rulesets.Osu/Edit/OsuGridToolboxGroup.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

311 lines
11 KiB
C#
Raw Normal View History

2023-12-29 05:36:30 +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;
using System.Linq;
2023-12-29 05:36:30 +08:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
2023-12-29 06:10:06 +08:00
using osu.Framework.Input.Bindings;
using osu.Framework.Input.Events;
using osu.Framework.Utils;
2024-06-06 00:12:02 +08:00
using osu.Game.Graphics.Containers;
2023-12-29 05:36:30 +08:00
using osu.Game.Graphics.UserInterface;
2023-12-29 06:10:06 +08:00
using osu.Game.Input.Bindings;
using osu.Game.Rulesets.Edit;
using osu.Game.Rulesets.Objects.Types;
2023-12-29 06:10:06 +08:00
using osu.Game.Rulesets.Osu.UI;
2023-12-29 05:36:30 +08:00
using osu.Game.Screens.Edit;
using osu.Game.Screens.Edit.Components.RadioButtons;
using osuTK;
using osuTK.Graphics;
2023-12-29 05:36:30 +08:00
2023-12-29 06:10:06 +08:00
namespace osu.Game.Rulesets.Osu.Edit
2023-12-29 05:36:30 +08:00
{
2023-12-29 06:10:06 +08:00
public partial class OsuGridToolboxGroup : EditorToolboxGroup, IKeyBindingHandler<GlobalAction>
2023-12-29 05:36:30 +08:00
{
[Resolved]
private EditorBeatmap editorBeatmap { get; set; } = null!;
2024-06-06 00:12:02 +08:00
[Resolved]
private IExpandingContainer? expandingContainer { get; set; }
/// <summary>
/// X position of the grid's origin.
/// </summary>
2023-12-29 06:10:06 +08:00
public BindableFloat StartPositionX { get; } = new BindableFloat(OsuPlayfield.BASE_SIZE.X / 2)
2023-12-29 05:36:30 +08:00
{
MinValue = 0f,
2023-12-29 06:10:06 +08:00
MaxValue = OsuPlayfield.BASE_SIZE.X,
2023-12-29 05:36:30 +08:00
Precision = 1f
};
/// <summary>
/// Y position of the grid's origin.
/// </summary>
2023-12-29 06:10:06 +08:00
public BindableFloat StartPositionY { get; } = new BindableFloat(OsuPlayfield.BASE_SIZE.Y / 2)
2023-12-29 05:36:30 +08:00
{
MinValue = 0f,
2023-12-29 06:10:06 +08:00
MaxValue = OsuPlayfield.BASE_SIZE.Y,
2023-12-29 05:36:30 +08:00
Precision = 1f
};
/// <summary>
/// The spacing between grid lines.
/// </summary>
2023-12-29 05:36:30 +08:00
public BindableFloat Spacing { get; } = new BindableFloat(4f)
{
MinValue = 4f,
MaxValue = 128f,
Precision = 1f
};
/// <summary>
/// Rotation of the grid lines in degrees.
/// </summary>
2023-12-29 05:36:30 +08:00
public BindableFloat GridLinesRotation { get; } = new BindableFloat(0f)
{
MinValue = -180f,
MaxValue = 180f,
2023-12-29 05:36:30 +08:00
Precision = 1f
};
/// <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>();
public Bindable<PositionSnapGridType> GridType { get; } = new Bindable<PositionSnapGridType>();
2023-12-29 05:36:30 +08:00
private ExpandableSlider<float> startPositionXSlider = null!;
private ExpandableSlider<float> startPositionYSlider = null!;
private ExpandableSlider<float> spacingSlider = null!;
private ExpandableSlider<float> gridLinesRotationSlider = null!;
private EditorRadioButtonCollection gridTypeButtons = null!;
2023-12-29 05:36:30 +08:00
private ExpandableButton useSelectedObjectPositionButton = null!;
2023-12-29 06:10:06 +08:00
public OsuGridToolboxGroup()
: base("grid")
{
}
private const float max_automatic_spacing = 64;
2023-12-29 05:36:30 +08:00
[BackgroundDependencyLoader]
private void load()
{
Children = new Drawable[]
{
startPositionXSlider = new ExpandableSlider<float>
{
2023-12-31 23:41:22 +08:00
Current = StartPositionX,
KeyboardStep = 1,
2023-12-29 05:36:30 +08:00
},
startPositionYSlider = new ExpandableSlider<float>
{
2023-12-31 23:41:22 +08:00
Current = StartPositionY,
KeyboardStep = 1,
2023-12-29 05:36:30 +08:00
},
useSelectedObjectPositionButton = new ExpandableButton
{
ExpandedLabelText = "Centre on selected object",
Action = () =>
{
if (editorBeatmap.SelectedHitObjects.Count != 1)
return;
var position = ((IHasPosition)editorBeatmap.SelectedHitObjects.Single()).Position;
StartPosition.Value = new Vector2(MathF.Round(position.X), MathF.Round(position.Y));
updateEnabledStates();
},
RelativeSizeAxes = Axes.X,
},
2023-12-29 05:36:30 +08:00
spacingSlider = new ExpandableSlider<float>
{
2023-12-31 23:41:22 +08:00
Current = Spacing,
KeyboardStep = 1,
2023-12-29 05:36:30 +08:00
},
gridLinesRotationSlider = new ExpandableSlider<float>
{
2023-12-31 23:41:22 +08:00
Current = GridLinesRotation,
KeyboardStep = 1,
2023-12-29 23:59:12 +08: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-29 05:36:30 +08:00
};
2023-12-29 06:10:06 +08:00
2023-12-30 21:32:20 +08:00
Spacing.Value = editorBeatmap.BeatmapInfo.GridSize;
2023-12-29 05:36:30 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
gridTypeButtons.Items.First().Select();
2023-12-29 05:36:30 +08:00
StartPositionX.BindValueChanged(x =>
{
startPositionXSlider.ContractedLabelText = $"X: {x.NewValue:N0}";
startPositionXSlider.ExpandedLabelText = $"X Offset: {x.NewValue:N0}";
StartPosition.Value = new Vector2(x.NewValue, StartPosition.Value.Y);
2023-12-29 05:36:30 +08:00
}, true);
StartPositionY.BindValueChanged(y =>
{
startPositionYSlider.ContractedLabelText = $"Y: {y.NewValue:N0}";
startPositionYSlider.ExpandedLabelText = $"Y Offset: {y.NewValue:N0}";
StartPosition.Value = new Vector2(StartPosition.Value.X, y.NewValue);
2023-12-29 05:36:30 +08:00
}, true);
StartPosition.BindValueChanged(pos =>
{
StartPositionX.Value = pos.NewValue.X;
StartPositionY.Value = pos.NewValue.Y;
updateEnabledStates();
});
2023-12-29 05:36:30 +08:00
Spacing.BindValueChanged(spacing =>
{
spacingSlider.ContractedLabelText = $"S: {spacing.NewValue:N0}";
spacingSlider.ExpandedLabelText = $"Spacing: {spacing.NewValue:N0}";
SpacingVector.Value = new Vector2(spacing.NewValue);
2023-12-30 21:32:20 +08:00
editorBeatmap.BeatmapInfo.GridSize = (int)spacing.NewValue;
2023-12-29 05:36:30 +08:00
}, true);
GridLinesRotation.BindValueChanged(rotation =>
{
gridLinesRotationSlider.ContractedLabelText = $"R: {rotation.NewValue:#,0.##}";
gridLinesRotationSlider.ExpandedLabelText = $"Rotation: {rotation.NewValue:#,0.##}";
}, true);
GridType.BindValueChanged(v =>
{
GridLinesRotation.Disabled = v.NewValue == PositionSnapGridType.Circle;
switch (v.NewValue)
{
case PositionSnapGridType.Square:
2024-06-20 03:10:30 +08:00
GridLinesRotation.Value = ((GridLinesRotation.Value + 405) % 90) - 45;
GridLinesRotation.MinValue = -45;
GridLinesRotation.MaxValue = 45;
break;
case PositionSnapGridType.Triangle:
2024-06-20 03:10:30 +08:00
GridLinesRotation.Value = ((GridLinesRotation.Value + 390) % 60) - 30;
GridLinesRotation.MinValue = -30;
GridLinesRotation.MaxValue = 30;
break;
}
}, true);
editorBeatmap.BeatmapReprocessed += updateEnabledStates;
editorBeatmap.SelectedHitObjects.BindCollectionChanged((_, _) => updateEnabledStates());
expandingContainer?.Expanded.BindValueChanged(v =>
{
gridTypeButtons.FadeTo(v.NewValue ? 1f : 0f, 500, Easing.OutQuint);
gridTypeButtons.BypassAutoSizeAxes = !v.NewValue ? Axes.Y : Axes.None;
updateEnabledStates();
}, true);
}
private void updateEnabledStates()
{
useSelectedObjectPositionButton.Enabled.Value = expandingContainer?.Expanded.Value == true
&& editorBeatmap.SelectedHitObjects.Count == 1
&& !Precision.AlmostEquals(StartPosition.Value, ((IHasPosition)editorBeatmap.SelectedHitObjects.Single()).Position, 0.5f);
2023-12-29 05:36:30 +08:00
}
2023-12-29 06:10:06 +08:00
private void nextGridSize()
2023-12-29 06:10:06 +08:00
{
Spacing.Value = Spacing.Value * 2 >= max_automatic_spacing ? Spacing.Value / 8 : Spacing.Value * 2;
2023-12-29 06:10:06 +08:00
}
public bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
{
switch (e.Action)
{
case GlobalAction.EditorCycleGridDisplayMode:
nextGridSize();
2023-12-29 06:10:06 +08:00
return true;
}
return false;
}
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
{
}
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 23:59:12 +08:00
}
2023-12-29 05:36:30 +08:00
}