1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-08 23:27:32 +08:00
osu-lazer/osu.Game.Rulesets.Taiko/UI/DrumTouchInputArea.cs

289 lines
11 KiB
C#
Raw Normal View History

// 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.Collections.Generic;
using System.Diagnostics;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Input.Bindings;
using osu.Framework.Input.Events;
using osu.Game.Graphics;
using osu.Game.Rulesets.Taiko.Configuration;
using osuTK;
2022-07-22 15:18:22 +08:00
using osuTK.Graphics;
namespace osu.Game.Rulesets.Taiko.UI
{
/// <summary>
2022-07-22 15:18:22 +08:00
/// An overlay that captures and displays osu!taiko mouse and touch input.
/// </summary>
2022-11-24 13:32:20 +08:00
public partial class DrumTouchInputArea : VisibilityContainer
{
2022-07-22 16:48:07 +08:00
// visibility state affects our child. we always want to handle input.
public override bool PropagatePositionalInputSubTree => true;
public override bool PropagateNonPositionalInputSubTree => true;
private KeyBindingContainer<TaikoAction> keyBindingContainer = null!;
private readonly Dictionary<object, TaikoAction> trackedActions = new Dictionary<object, TaikoAction>();
2022-07-22 16:17:38 +08:00
private Container mainContent = null!;
2022-07-22 16:48:07 +08:00
private QuarterCircle leftCentre = null!;
private QuarterCircle rightCentre = null!;
private QuarterCircle leftRim = null!;
private QuarterCircle rightRim = null!;
private readonly Bindable<TaikoTouchControlScheme> configTouchControlScheme = new Bindable<TaikoTouchControlScheme>();
2023-01-10 23:08:18 +08:00
[Resolved]
private OsuColour colours { get; set; } = null!;
2022-07-22 16:17:38 +08:00
[BackgroundDependencyLoader]
2023-01-10 21:07:07 +08:00
private void load(TaikoInputManager taikoInputManager, TaikoRulesetConfigManager config)
2022-03-12 21:01:40 +08:00
{
2022-07-22 16:17:38 +08:00
Debug.Assert(taikoInputManager.KeyBindingContainer != null);
keyBindingContainer = taikoInputManager.KeyBindingContainer;
2022-07-22 15:18:22 +08:00
2022-07-22 16:17:38 +08:00
// Container should handle input everywhere.
RelativeSizeAxes = Axes.Both;
2022-07-22 15:18:22 +08:00
2022-07-22 16:48:07 +08:00
const float centre_region = 0.80f;
config.BindWith(TaikoRulesetSetting.TouchControlScheme, configTouchControlScheme);
Children = new Drawable[]
{
2022-07-22 16:17:38 +08:00
new Container
2022-06-02 13:36:07 +08:00
{
2022-07-22 16:17:38 +08:00
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
RelativeSizeAxes = Axes.X,
2022-07-22 16:48:07 +08:00
Height = 350,
2022-07-22 16:17:38 +08:00
Y = 20,
Masking = true,
FillMode = FillMode.Fit,
Children = new Drawable[]
{
2022-07-22 16:17:38 +08:00
mainContent = new Container
2022-06-02 13:36:07 +08:00
{
RelativeSizeAxes = Axes.Both,
2022-07-22 16:17:38 +08:00
Children = new Drawable[]
{
leftRim = new QuarterCircle(getTaikoActionFromInput(TaikoAction.LeftRim), getColourFromTaikoAction(getTaikoActionFromInput(TaikoAction.LeftRim)))
2022-07-22 16:17:38 +08:00
{
2022-07-22 16:48:07 +08:00
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomRight,
X = -2,
2022-07-22 16:17:38 +08:00
},
rightRim = new QuarterCircle(getTaikoActionFromInput(TaikoAction.RightRim), getColourFromTaikoAction(getTaikoActionFromInput(TaikoAction.RightRim)))
2022-07-22 16:17:38 +08:00
{
2022-07-22 16:48:07 +08:00
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomRight,
X = 2,
Rotation = 90,
2022-07-22 16:17:38 +08:00
},
leftCentre = new QuarterCircle(getTaikoActionFromInput(TaikoAction.LeftCentre), getColourFromTaikoAction(getTaikoActionFromInput(TaikoAction.LeftCentre)))
2022-07-22 16:17:38 +08:00
{
Anchor = Anchor.BottomCentre,
2022-07-22 16:48:07 +08:00
Origin = Anchor.BottomRight,
X = -2,
Scale = new Vector2(centre_region),
2022-07-22 16:17:38 +08:00
},
rightCentre = new QuarterCircle(getTaikoActionFromInput(TaikoAction.RightCentre), getColourFromTaikoAction(getTaikoActionFromInput(TaikoAction.RightCentre)))
2022-07-22 16:48:07 +08:00
{
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomRight,
X = 2,
Scale = new Vector2(centre_region),
Rotation = 90,
}
2022-07-22 16:17:38 +08:00
}
},
}
},
};
}
protected override bool OnKeyDown(KeyDownEvent e)
{
// Hide whenever the keyboard is used.
2022-07-22 16:48:07 +08:00
Hide();
return false;
}
2022-07-22 15:18:22 +08:00
protected override bool OnTouchDown(TouchDownEvent e)
{
handleDown(e.Touch.Source, e.ScreenSpaceTouchDownPosition);
return true;
}
protected override void OnTouchUp(TouchUpEvent e)
{
handleUp(e.Touch.Source);
base.OnTouchUp(e);
}
private void handleDown(object source, Vector2 position)
{
2022-07-22 16:48:07 +08:00
Show();
TaikoAction TaikoAction = getTaikoActionFromPosition(position);
// Not too sure how this can happen, but let's avoid throwing.
if (trackedActions.ContainsKey(source))
return;
trackedActions.Add(source, TaikoAction);
keyBindingContainer.TriggerPressed(TaikoAction);
}
private void handleUp(object source)
{
keyBindingContainer.TriggerReleased(trackedActions[source]);
trackedActions.Remove(source);
}
private bool validMouse(MouseButtonEvent e) =>
leftRim.Contains(e.ScreenSpaceMouseDownPosition) || rightRim.Contains(e.ScreenSpaceMouseDownPosition);
private TaikoAction getTaikoActionFromInput(TaikoAction input)
{
switch (configTouchControlScheme.Value)
{
case TaikoTouchControlScheme.KDDK:
switch (input)
{
case TaikoAction.LeftRim: return TaikoAction.LeftRim;
case TaikoAction.LeftCentre: return TaikoAction.LeftCentre;
case TaikoAction.RightCentre: return TaikoAction.RightCentre;
case TaikoAction.RightRim: return TaikoAction.RightRim;
}
break;
case TaikoTouchControlScheme.DDKK:
switch (input)
{
case TaikoAction.LeftRim: return TaikoAction.LeftCentre;
case TaikoAction.LeftCentre: return TaikoAction.RightCentre;
case TaikoAction.RightCentre: return TaikoAction.LeftRim;
case TaikoAction.RightRim: return TaikoAction.RightRim;
}
break;
case TaikoTouchControlScheme.KKDD:
switch (input)
{
case TaikoAction.LeftRim: return TaikoAction.LeftRim;
case TaikoAction.LeftCentre: return TaikoAction.RightRim;
case TaikoAction.RightCentre: return TaikoAction.LeftCentre;
case TaikoAction.RightRim: return TaikoAction.RightCentre;
}
break;
}
return TaikoAction.LeftCentre;
}
private TaikoAction getTaikoActionFromPosition(Vector2 inputPosition)
2022-03-12 21:01:40 +08:00
{
2022-07-22 16:48:07 +08:00
bool centreHit = leftCentre.Contains(inputPosition) || rightCentre.Contains(inputPosition);
2022-07-22 15:18:22 +08:00
bool leftSide = ToLocalSpace(inputPosition).X < DrawWidth / 2;
TaikoAction input;
2022-07-22 15:18:22 +08:00
if (leftSide)
input = centreHit ? TaikoAction.LeftCentre : TaikoAction.LeftRim;
else
input = centreHit ? TaikoAction.RightCentre : TaikoAction.RightRim;
return getTaikoActionFromInput(input);
}
2022-07-22 16:48:07 +08:00
protected override void PopIn()
{
mainContent.FadeIn(500, Easing.OutQuint);
}
protected override void PopOut()
{
mainContent.FadeOut(300);
}
2023-01-10 23:08:18 +08:00
private Color4 getColourFromTaikoAction(TaikoAction handledAction)
{
switch (handledAction)
{
case TaikoAction.LeftRim:
case TaikoAction.RightRim:
return colours.Blue;
case TaikoAction.LeftCentre:
case TaikoAction.RightCentre:
return colours.Red;
2023-01-10 23:08:18 +08:00
}
return colours.Red;
}
2022-11-24 13:32:20 +08:00
private partial class QuarterCircle : CompositeDrawable, IKeyBindingHandler<TaikoAction>
2022-07-22 16:48:07 +08:00
{
private readonly Circle overlay;
private readonly TaikoAction handledAction;
private readonly Circle circle;
public override bool Contains(Vector2 screenSpacePos) => circle.Contains(screenSpacePos);
2023-01-10 23:08:18 +08:00
public QuarterCircle(TaikoAction handledAction, Color4 colour)
2022-07-22 16:48:07 +08:00
{
this.handledAction = handledAction;
RelativeSizeAxes = Axes.Both;
FillMode = FillMode.Fit;
InternalChildren = new Drawable[]
{
new Container
{
Masking = true,
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
circle = new Circle
{
RelativeSizeAxes = Axes.Both,
Colour = colour.Multiply(1.4f).Darken(2.8f),
2022-07-22 16:48:07 +08:00
Alpha = 0.8f,
Scale = new Vector2(2),
},
overlay = new Circle
{
Alpha = 0,
RelativeSizeAxes = Axes.Both,
Blending = BlendingParameters.Additive,
Colour = colour,
Scale = new Vector2(2),
}
}
},
};
}
public bool OnPressed(KeyBindingPressEvent<TaikoAction> e)
{
if (e.Action == handledAction)
overlay.FadeTo(1f, 80, Easing.OutQuint);
2022-07-22 16:48:07 +08:00
return false;
}
public void OnReleased(KeyBindingReleaseEvent<TaikoAction> e)
{
if (e.Action == handledAction)
overlay.FadeOut(1000, Easing.OutQuint);
}
}
}
}