1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-22 01:27:29 +08:00
osu-lazer/osu.Game.Rulesets.Taiko/UI/DrumTouchInputArea.cs

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

299 lines
10 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;
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-07-22 16:48:07 +08:00
public partial class DrumTouchInputArea : VisibilityContainer
{
public TaikoTouchControlScheme? ForceControlScheme { get; set; }
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 static 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;
if (ForceControlScheme == null)
config.BindWith(TaikoRulesetSetting.TouchControlScheme, configTouchControlScheme);
else
configTouchControlScheme.Value = ForceControlScheme.Value;
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(0, colours)
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
},
leftCentre = new QuarterCircle(1, colours)
2022-07-22 16:17:38 +08:00
{
2022-07-22 16:48:07 +08:00
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomRight,
X = -2,
Scale = new Vector2(centre_region),
2022-07-22 16:17:38 +08:00
},
rightRim = new QuarterCircle(3, colours)
2022-07-22 16:17:38 +08:00
{
Anchor = Anchor.BottomCentre,
2022-07-22 16:48:07 +08:00
Origin = Anchor.BottomRight,
X = 2,
Rotation = 90,
2022-07-22 16:17:38 +08:00
},
rightCentre = new QuarterCircle(2, colours)
2022-07-22 16:48:07 +08:00
{
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomRight,
X = 2,
Scale = new Vector2(centre_region),
2022-07-22 16:48:07 +08:00
Rotation = 90,
2023-01-11 04:05:41 +08:00
}
2022-07-22 16:17:38 +08:00
}
},
}
},
};
}
private static readonly TaikoAction[,] mappedTaikoAction =
2023-01-12 04:06:43 +08:00
{
{
// KDDK
TaikoAction.LeftRim,
TaikoAction.LeftCentre,
TaikoAction.RightCentre,
TaikoAction.RightRim
},
2023-01-12 04:06:43 +08:00
{
// DDKK
TaikoAction.LeftCentre,
TaikoAction.RightCentre,
TaikoAction.LeftRim,
TaikoAction.RightRim
},
2023-01-12 04:06:43 +08:00
{
// KKDD
TaikoAction.LeftRim,
TaikoAction.RightRim,
TaikoAction.LeftCentre,
TaikoAction.RightCentre
}
};
private static TaikoAction getTaikoActionFromDrumSegment(int drumSegment)
{
return mappedTaikoAction[(int)configTouchControlScheme.Value, drumSegment];
}
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 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;
int drumSegment;
2022-07-22 15:18:22 +08:00
if (leftSide)
drumSegment = centreHit ? 1 : 0;
else
drumSegment = centreHit ? 2 : 3;
return getTaikoActionFromDrumSegment(drumSegment);
}
2022-07-22 16:48:07 +08:00
protected override void PopIn()
{
mainContent.FadeIn(500, Easing.OutQuint);
}
protected override void PopOut()
{
mainContent.FadeOut(300);
}
private partial class QuarterCircle : CompositeDrawable, IKeyBindingHandler<TaikoAction>
{
private readonly Circle overlay;
private readonly int drumSegment;
private readonly TaikoAction taikoAction;
private readonly OsuColour colours;
private readonly Color4 colour;
2022-07-22 16:48:07 +08:00
private readonly Circle circle;
public override bool Contains(Vector2 screenSpacePos) => circle.Contains(screenSpacePos);
public QuarterCircle(int drumSegment, OsuColour colours)
2022-07-22 16:48:07 +08:00
{
this.drumSegment = drumSegment;
this.colours = colours;
2022-07-22 16:48:07 +08:00
RelativeSizeAxes = Axes.Both;
FillMode = FillMode.Fit;
taikoAction = getTaikoActionFromDrumSegment(drumSegment);
colour = getColorFromTaikoAction(taikoAction);
2022-07-22 16:48:07 +08:00
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),
}
}
},
};
}
private Color4 getColorFromTaikoAction(TaikoAction handledAction)
{
switch (handledAction)
{
case TaikoAction.LeftRim:
case TaikoAction.RightRim:
return colours.Blue;
case TaikoAction.LeftCentre:
case TaikoAction.RightCentre:
return colours.Red;
}
throw new ArgumentOutOfRangeException();
}
2022-07-22 16:48:07 +08:00
public bool OnPressed(KeyBindingPressEvent<TaikoAction> e)
{
if (e.Action == taikoAction)
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 == taikoAction)
2022-07-22 16:48:07 +08:00
overlay.FadeOut(1000, Easing.OutQuint);
}
}
}
}