osu/osu.Game.Rulesets.Catch/Edit/CatchHitObjectComposer.cs

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

224 lines
8.1 KiB
C#
Raw Normal View History

2021-06-22 01:05:29 +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.
2022-06-17 07:37:17 +00:00
#nullable disable
using System;
2021-06-22 01:05:29 +00:00
using System.Collections.Generic;
2021-10-25 07:37:44 +00:00
using System.Linq;
using JetBrains.Annotations;
2021-07-07 07:10:24 +00:00
using osu.Framework.Allocation;
2021-10-25 07:37:44 +00:00
using osu.Framework.Bindables;
using osu.Framework.Extensions.EnumExtensions;
2021-07-07 07:10:24 +00:00
using osu.Framework.Graphics;
2021-10-25 07:37:44 +00:00
using osu.Framework.Input;
using osu.Framework.Input.Events;
using osu.Game.Beatmaps;
2021-10-25 07:37:44 +00:00
using osu.Game.Graphics.UserInterface;
using osu.Game.Input.Bindings;
2021-06-22 01:05:29 +00:00
using osu.Game.Rulesets.Catch.Objects;
2021-10-25 07:37:44 +00:00
using osu.Game.Rulesets.Catch.UI;
2021-06-22 01:05:29 +00:00
using osu.Game.Rulesets.Edit;
using osu.Game.Rulesets.Edit.Tools;
using osu.Game.Rulesets.Mods;
2021-10-25 07:37:44 +00:00
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.UI;
using osu.Game.Screens.Edit.Compose.Components;
using osuTK;
2021-06-22 01:05:29 +00:00
namespace osu.Game.Rulesets.Catch.Edit
{
public partial class CatchHitObjectComposer : DistancedHitObjectComposer<CatchHitObject>
2021-06-22 01:05:29 +00:00
{
2021-10-25 07:37:44 +00:00
private const float distance_snap_radius = 50;
private CatchDistanceSnapGrid distanceSnapGrid;
private InputManager inputManager;
private readonly BindableDouble timeRangeMultiplier = new BindableDouble(1)
{
MinValue = 1,
MaxValue = 10,
};
2021-06-22 01:05:29 +00:00
public CatchHitObjectComposer(CatchRuleset ruleset)
: base(ruleset)
{
}
2021-07-07 07:10:24 +00:00
[BackgroundDependencyLoader]
private void load()
{
2022-04-28 07:57:14 +00:00
// todo: enable distance spacing once catch supports applying it to its existing distance snap grid implementation.
RightSideToolboxContainer.Alpha = 0;
DistanceSpacingMultiplier.Disabled = true;
2021-07-07 07:10:24 +00:00
LayerBelowRuleset.Add(new PlayfieldBorder
{
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
RelativeSizeAxes = Axes.X,
Height = CatchPlayfield.HEIGHT,
2021-07-07 07:10:24 +00:00
PlayfieldBorderStyle = { Value = PlayfieldBorderStyle.Corners }
});
2021-10-25 07:37:44 +00:00
LayerBelowRuleset.Add(distanceSnapGrid = new CatchDistanceSnapGrid(new[]
{
0.0,
Catcher.BASE_DASH_SPEED, -Catcher.BASE_DASH_SPEED,
Catcher.BASE_WALK_SPEED, -Catcher.BASE_WALK_SPEED,
2021-10-25 07:37:44 +00:00
}));
}
protected override void LoadComplete()
{
base.LoadComplete();
inputManager = GetContainingInputManager();
}
protected override double ReadCurrentDistanceSnap(HitObject before, HitObject after)
{
// osu!catch's distance snap implementation is limited, in that a custom spacing cannot be specified.
// Therefore this functionality is not currently used.
//
// The implementation below is probably correct but should be checked if/when exposed via controls.
float expectedDistance = DurationToDistance(before, after.StartTime - before.GetEndTime());
float actualDistance = Math.Abs(((CatchHitObject)before).EffectiveX - ((CatchHitObject)after).EffectiveX);
return actualDistance / expectedDistance;
}
2021-10-25 07:37:44 +00:00
protected override void Update()
{
base.Update();
updateDistanceSnapGrid();
2021-07-07 07:10:24 +00:00
}
public override bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
{
switch (e.Action)
{
// Note that right now these are hard to use as the default key bindings conflict with existing editor key bindings.
// In the future we will want to expose this via UI and potentially change the key bindings to be editor-specific.
// May be worth considering standardising "zoom" behaviour with what the timeline uses (ie. alt-wheel) but that may cause new conflicts.
case GlobalAction.IncreaseScrollSpeed:
this.TransformBindableTo(timeRangeMultiplier, timeRangeMultiplier.Value - 1, 200, Easing.OutQuint);
break;
case GlobalAction.DecreaseScrollSpeed:
this.TransformBindableTo(timeRangeMultiplier, timeRangeMultiplier.Value + 1, 200, Easing.OutQuint);
break;
}
return base.OnPressed(e);
}
protected override DrawableRuleset<CatchHitObject> CreateDrawableRuleset(Ruleset ruleset, IBeatmap beatmap, IReadOnlyList<Mod> mods = null) =>
new DrawableCatchEditorRuleset(ruleset, beatmap, mods)
{
TimeRangeMultiplier = { BindTarget = timeRangeMultiplier, }
};
2021-06-22 03:10:16 +00:00
protected override IReadOnlyList<HitObjectCompositionTool> CompositionTools => new HitObjectCompositionTool[]
{
new FruitCompositionTool(),
new JuiceStreamCompositionTool(),
2021-06-22 03:30:09 +00:00
new BananaShowerCompositionTool()
2021-06-22 03:10:16 +00:00
};
public override SnapResult FindSnappedPositionAndTime(Vector2 screenSpacePosition, SnapType snapType = SnapType.All)
{
var result = base.FindSnappedPositionAndTime(screenSpacePosition, snapType);
result.ScreenSpacePosition.X = screenSpacePosition.X;
2021-10-25 07:37:44 +00:00
if (snapType.HasFlagFast(SnapType.Grids))
2021-10-25 07:37:44 +00:00
{
if (distanceSnapGrid.IsPresent && distanceSnapGrid.GetSnappedPosition(result.ScreenSpacePosition) is SnapResult snapResult &&
Vector2.Distance(snapResult.ScreenSpacePosition, result.ScreenSpacePosition) < distance_snap_radius)
{
result = snapResult;
}
2021-10-25 07:37:44 +00:00
}
return result;
}
protected override ComposeBlueprintContainer CreateBlueprintContainer() => new CatchBlueprintContainer(this);
2021-10-25 07:37:44 +00:00
[CanBeNull]
2021-10-26 02:14:12 +00:00
private PalpableCatchHitObject getLastSnappableHitObject(double time)
2021-10-25 07:37:44 +00:00
{
var hitObject = EditorBeatmap.HitObjects.OfType<CatchHitObject>().LastOrDefault(h => h.GetEndTime() < time && !(h is BananaShower));
switch (hitObject)
{
case Fruit fruit:
return fruit;
case JuiceStream juiceStream:
return juiceStream.NestedHitObjects.OfType<PalpableCatchHitObject>().LastOrDefault(h => !(h is TinyDroplet));
default:
return null;
}
}
[CanBeNull]
private PalpableCatchHitObject getDistanceSnapGridSourceHitObject()
{
switch (BlueprintContainer.CurrentTool)
{
2022-06-24 12:25:23 +00:00
case SelectTool:
2021-10-25 07:37:44 +00:00
if (EditorBeatmap.SelectedHitObjects.Count == 0)
return null;
double minTime = EditorBeatmap.SelectedHitObjects.Min(hitObject => hitObject.StartTime);
2021-10-26 02:14:12 +00:00
return getLastSnappableHitObject(minTime);
2021-10-25 07:37:44 +00:00
2022-06-24 12:25:23 +00:00
case FruitCompositionTool:
case JuiceStreamCompositionTool:
2021-10-25 07:37:44 +00:00
if (!CursorInPlacementArea)
return null;
if (EditorBeatmap.PlacementObject.Value is JuiceStream)
{
// Juice stream path is not subject to snapping.
return null;
}
double timeAtCursor = ((CatchPlayfield)Playfield).TimeAtScreenSpacePosition(inputManager.CurrentState.Mouse.Position);
2021-10-26 02:14:12 +00:00
return getLastSnappableHitObject(timeAtCursor);
2021-10-25 07:37:44 +00:00
default:
return null;
}
}
private void updateDistanceSnapGrid()
{
if (DistanceSnapToggle.Value != TernaryState.True)
2021-10-25 07:37:44 +00:00
{
distanceSnapGrid.Hide();
return;
2021-10-25 07:37:44 +00:00
}
var sourceHitObject = getDistanceSnapGridSourceHitObject();
if (sourceHitObject == null)
2021-10-25 07:37:44 +00:00
{
distanceSnapGrid.Hide();
return;
2021-10-25 07:37:44 +00:00
}
distanceSnapGrid.Show();
distanceSnapGrid.StartTime = sourceHitObject.GetEndTime();
distanceSnapGrid.StartX = sourceHitObject.EffectiveX;
2021-10-25 07:37:44 +00:00
}
2021-06-22 01:05:29 +00:00
}
}