mirror of
https://github.com/ppy/osu.git
synced 2025-01-22 08:32:54 +08:00
195 lines
5.8 KiB
C#
195 lines
5.8 KiB
C#
// 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.Graphics;
|
|
using osu.Framework.Input;
|
|
using osu.Framework.Input.Events;
|
|
using osu.Game.Graphics;
|
|
using osu.Game.Rulesets.Edit;
|
|
using osu.Game.Rulesets.Objects;
|
|
using osu.Game.Rulesets.Objects.Types;
|
|
using osu.Game.Rulesets.Osu.Edit.Blueprints.HitCircles.Components;
|
|
using osu.Game.Rulesets.Osu.Edit.Blueprints.Sliders.Components;
|
|
using osuTK;
|
|
using osuTK.Input;
|
|
|
|
namespace osu.Game.Rulesets.Osu.Edit.Blueprints.Sliders
|
|
{
|
|
public class SliderPlacementBlueprint : PlacementBlueprint
|
|
{
|
|
public new Objects.Slider HitObject => (Objects.Slider)base.HitObject;
|
|
|
|
private SliderBodyPiece bodyPiece;
|
|
private HitCirclePiece headCirclePiece;
|
|
private HitCirclePiece tailCirclePiece;
|
|
|
|
private InputManager inputManager;
|
|
|
|
private PlacementState state;
|
|
private PathControlPoint segmentStart;
|
|
private PathControlPoint cursor;
|
|
private int currentSegmentLength;
|
|
|
|
[Resolved(CanBeNull = true)]
|
|
private HitObjectComposer composer { get; set; }
|
|
|
|
public SliderPlacementBlueprint()
|
|
: base(new Objects.Slider())
|
|
{
|
|
RelativeSizeAxes = Axes.Both;
|
|
|
|
HitObject.Path.ControlPoints.Add(segmentStart = new PathControlPoint(Vector2.Zero, PathType.Linear));
|
|
currentSegmentLength = 1;
|
|
}
|
|
|
|
[BackgroundDependencyLoader]
|
|
private void load(OsuColour colours)
|
|
{
|
|
InternalChildren = new Drawable[]
|
|
{
|
|
bodyPiece = new SliderBodyPiece(),
|
|
headCirclePiece = new HitCirclePiece(),
|
|
tailCirclePiece = new HitCirclePiece(),
|
|
new PathControlPointVisualiser(HitObject, false)
|
|
};
|
|
|
|
setState(PlacementState.Initial);
|
|
}
|
|
|
|
protected override void LoadComplete()
|
|
{
|
|
base.LoadComplete();
|
|
inputManager = GetContainingInputManager();
|
|
}
|
|
|
|
public override void UpdatePosition(Vector2 screenSpacePosition)
|
|
{
|
|
switch (state)
|
|
{
|
|
case PlacementState.Initial:
|
|
HitObject.Position = ToLocalSpace(screenSpacePosition);
|
|
break;
|
|
|
|
case PlacementState.Body:
|
|
ensureCursor();
|
|
|
|
// The given screen-space position may have been externally snapped, but the unsnapped position from the input manager
|
|
// is used instead since snapping control points doesn't make much sense
|
|
cursor.Position.Value = ToLocalSpace(inputManager.CurrentState.Mouse.Position) - HitObject.Position;
|
|
break;
|
|
}
|
|
}
|
|
|
|
protected override bool OnClick(ClickEvent e)
|
|
{
|
|
switch (state)
|
|
{
|
|
case PlacementState.Initial:
|
|
beginCurve();
|
|
break;
|
|
|
|
case PlacementState.Body:
|
|
switch (e.Button)
|
|
{
|
|
case MouseButton.Left:
|
|
ensureCursor();
|
|
|
|
// Detatch the cursor
|
|
cursor = null;
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
protected override void OnMouseUp(MouseUpEvent e)
|
|
{
|
|
if (state == PlacementState.Body && e.Button == MouseButton.Right)
|
|
endCurve();
|
|
base.OnMouseUp(e);
|
|
}
|
|
|
|
protected override bool OnDoubleClick(DoubleClickEvent e)
|
|
{
|
|
// Todo: This should all not occur on double click, but rather if the previous control point is hovered.
|
|
segmentStart = HitObject.Path.ControlPoints[^1];
|
|
segmentStart.Type.Value = PathType.Linear;
|
|
|
|
currentSegmentLength = 1;
|
|
return true;
|
|
}
|
|
|
|
private void beginCurve()
|
|
{
|
|
BeginPlacement();
|
|
setState(PlacementState.Body);
|
|
}
|
|
|
|
private void endCurve()
|
|
{
|
|
updateSlider();
|
|
EndPlacement();
|
|
}
|
|
|
|
protected override void Update()
|
|
{
|
|
base.Update();
|
|
updateSlider();
|
|
}
|
|
|
|
private void updatePathType()
|
|
{
|
|
switch (currentSegmentLength)
|
|
{
|
|
case 1:
|
|
case 2:
|
|
segmentStart.Type.Value = PathType.Linear;
|
|
break;
|
|
|
|
case 3:
|
|
segmentStart.Type.Value = PathType.PerfectCurve;
|
|
break;
|
|
|
|
default:
|
|
segmentStart.Type.Value = PathType.Bezier;
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void ensureCursor()
|
|
{
|
|
if (cursor == null)
|
|
{
|
|
HitObject.Path.ControlPoints.Add(cursor = new PathControlPoint { Position = { Value = Vector2.Zero } });
|
|
currentSegmentLength++;
|
|
|
|
updatePathType();
|
|
}
|
|
}
|
|
|
|
private void updateSlider()
|
|
{
|
|
HitObject.Path.ExpectedDistance.Value = composer?.GetSnappedDistanceFromDistance(HitObject.StartTime, (float)HitObject.Path.CalculatedDistance) ?? (float)HitObject.Path.CalculatedDistance;
|
|
|
|
bodyPiece.UpdateFrom(HitObject);
|
|
headCirclePiece.UpdateFrom(HitObject.HeadCircle);
|
|
tailCirclePiece.UpdateFrom(HitObject.TailCircle);
|
|
}
|
|
|
|
private void setState(PlacementState newState)
|
|
{
|
|
state = newState;
|
|
}
|
|
|
|
private enum PlacementState
|
|
{
|
|
Initial,
|
|
Body,
|
|
}
|
|
}
|
|
}
|