2022-04-28 10:49:37 +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 osu.Framework.Allocation;
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
using osu.Framework.Input.Events;
|
|
|
|
using osu.Game.Graphics.Containers;
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
|
|
|
using osu.Game.Overlays.Settings.Sections;
|
|
|
|
using osu.Game.Rulesets.Objects;
|
|
|
|
using osuTK;
|
|
|
|
|
|
|
|
namespace osu.Game.Rulesets.Edit
|
|
|
|
{
|
|
|
|
/// <summary>
|
|
|
|
/// Represents a <see cref="HitObjectComposer{TObject}"/> for rulesets with the concept of distances between objects.
|
|
|
|
/// </summary>
|
|
|
|
/// <typeparam name="TObject">The base type of supported objects.</typeparam>
|
|
|
|
[Cached(typeof(IDistanceSnapProvider))]
|
|
|
|
public abstract class DistancedHitObjectComposer<TObject> : HitObjectComposer<TObject>, IDistanceSnapProvider
|
|
|
|
where TObject : HitObject
|
|
|
|
{
|
2022-04-28 15:24:36 +08:00
|
|
|
protected Bindable<float> DistanceSpacingMultiplier { get; } = new BindableFloat(1.0f)
|
2022-04-28 10:49:37 +08:00
|
|
|
{
|
2022-04-28 15:24:36 +08:00
|
|
|
MinValue = 0.1f,
|
|
|
|
MaxValue = 6.0f,
|
|
|
|
Precision = 0.01f,
|
2022-04-28 10:49:37 +08:00
|
|
|
};
|
|
|
|
|
2022-04-28 15:24:36 +08:00
|
|
|
IBindable<float> IDistanceSnapProvider.DistanceSpacingMultiplier => DistanceSpacingMultiplier;
|
2022-04-28 10:49:37 +08:00
|
|
|
|
|
|
|
protected ExpandingToolboxContainer RightSideToolboxContainer { get; private set; }
|
|
|
|
|
2022-04-28 15:24:36 +08:00
|
|
|
private ExpandableSlider<float, SizeSlider> distanceSpacingSlider;
|
2022-04-28 10:49:37 +08:00
|
|
|
private bool distanceSpacingScrollActive;
|
|
|
|
|
|
|
|
protected DistancedHitObjectComposer(Ruleset ruleset)
|
|
|
|
: base(ruleset)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
private void load()
|
|
|
|
{
|
|
|
|
AddInternal(RightSideToolboxContainer = new ExpandingToolboxContainer
|
|
|
|
{
|
2022-04-28 10:50:55 +08:00
|
|
|
Alpha = DistanceSpacingMultiplier.Disabled ? 0 : 1,
|
2022-04-28 10:49:37 +08:00
|
|
|
Anchor = Anchor.TopRight,
|
|
|
|
Origin = Anchor.TopRight,
|
|
|
|
Child = new EditorToolboxGroup("snapping")
|
|
|
|
{
|
2022-04-28 15:24:36 +08:00
|
|
|
Child = distanceSpacingSlider = new ExpandableSlider<float, SizeSlider>
|
2022-04-28 10:49:37 +08:00
|
|
|
{
|
|
|
|
Current = { BindTarget = DistanceSpacingMultiplier },
|
|
|
|
KeyboardStep = 0.1f,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void LoadComplete()
|
|
|
|
{
|
|
|
|
base.LoadComplete();
|
|
|
|
|
2022-04-28 10:50:55 +08:00
|
|
|
if (!DistanceSpacingMultiplier.Disabled)
|
2022-04-28 10:49:37 +08:00
|
|
|
{
|
2022-04-28 10:50:55 +08:00
|
|
|
DistanceSpacingMultiplier.Value = EditorBeatmap.BeatmapInfo.DistanceSpacing;
|
|
|
|
DistanceSpacingMultiplier.BindValueChanged(v =>
|
|
|
|
{
|
|
|
|
distanceSpacingSlider.ContractedLabelText = $"D. S. ({v.NewValue:0.##x})";
|
|
|
|
distanceSpacingSlider.ExpandedLabelText = $"Distance Spacing ({v.NewValue:0.##x})";
|
|
|
|
EditorBeatmap.BeatmapInfo.DistanceSpacing = v.NewValue;
|
|
|
|
}, true);
|
|
|
|
}
|
2022-04-28 10:49:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected override bool OnKeyDown(KeyDownEvent e)
|
|
|
|
{
|
2022-04-28 10:50:55 +08:00
|
|
|
if (!DistanceSpacingMultiplier.Disabled && e.ControlPressed && e.AltPressed && !e.Repeat)
|
2022-04-28 10:49:37 +08:00
|
|
|
{
|
|
|
|
RightSideToolboxContainer.Expanded.Value = true;
|
|
|
|
distanceSpacingScrollActive = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return base.OnKeyDown(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void OnKeyUp(KeyUpEvent e)
|
|
|
|
{
|
2022-04-28 10:50:55 +08:00
|
|
|
if (!DistanceSpacingMultiplier.Disabled && distanceSpacingScrollActive && (!e.AltPressed || !e.ControlPressed))
|
2022-04-28 10:49:37 +08:00
|
|
|
{
|
|
|
|
RightSideToolboxContainer.Expanded.Value = false;
|
|
|
|
distanceSpacingScrollActive = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override bool OnScroll(ScrollEvent e)
|
|
|
|
{
|
|
|
|
if (distanceSpacingScrollActive)
|
|
|
|
{
|
|
|
|
DistanceSpacingMultiplier.Value += e.ScrollDelta.Y * (e.IsPrecise ? 0.01f : 0.1f);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return base.OnScroll(e);
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual float GetBeatSnapDistanceAt(HitObject referenceObject)
|
|
|
|
{
|
2022-04-28 15:54:38 +08:00
|
|
|
return (float)(100 * EditorBeatmap.Difficulty.SliderMultiplier * referenceObject.DifficultyControlPoint.SliderVelocity / BeatSnapProvider.BeatDivisor);
|
2022-04-28 10:49:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public virtual float DurationToDistance(HitObject referenceObject, double duration)
|
|
|
|
{
|
|
|
|
double beatLength = BeatSnapProvider.GetBeatLengthAtTime(referenceObject.StartTime);
|
|
|
|
return (float)(duration / beatLength * GetBeatSnapDistanceAt(referenceObject));
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual double DistanceToDuration(HitObject referenceObject, float distance)
|
|
|
|
{
|
|
|
|
double beatLength = BeatSnapProvider.GetBeatLengthAtTime(referenceObject.StartTime);
|
|
|
|
return distance / GetBeatSnapDistanceAt(referenceObject) * beatLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
public virtual double GetSnappedDurationFromDistance(HitObject referenceObject, float distance)
|
|
|
|
=> BeatSnapProvider.SnapTime(referenceObject.StartTime + DistanceToDuration(referenceObject, distance), referenceObject.StartTime) - referenceObject.StartTime;
|
|
|
|
|
|
|
|
public virtual float GetSnappedDistanceFromDistance(HitObject referenceObject, float distance)
|
|
|
|
{
|
|
|
|
double startTime = referenceObject.StartTime;
|
|
|
|
|
|
|
|
double actualDuration = startTime + DistanceToDuration(referenceObject, distance);
|
|
|
|
|
|
|
|
double snappedEndTime = BeatSnapProvider.SnapTime(actualDuration, startTime);
|
|
|
|
|
|
|
|
double beatLength = BeatSnapProvider.GetBeatLengthAtTime(startTime);
|
|
|
|
|
|
|
|
// we don't want to exceed the actual duration and snap to a point in the future.
|
|
|
|
// as we are snapping to beat length via SnapTime (which will round-to-nearest), check for snapping in the forward direction and reverse it.
|
|
|
|
if (snappedEndTime > actualDuration + 1)
|
|
|
|
snappedEndTime -= beatLength;
|
|
|
|
|
|
|
|
return DurationToDistance(referenceObject, snappedEndTime - startTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected class ExpandingToolboxContainer : ExpandingContainer
|
|
|
|
{
|
|
|
|
protected override double HoverExpansionDelay => 250;
|
|
|
|
|
|
|
|
public ExpandingToolboxContainer()
|
|
|
|
: base(130, 250)
|
|
|
|
{
|
|
|
|
RelativeSizeAxes = Axes.Y;
|
|
|
|
Padding = new MarginPadding { Left = 10 };
|
|
|
|
|
|
|
|
FillFlow.Spacing = new Vector2(10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|