1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-16 20:32:55 +08:00
osu-lazer/osu.Game/Screens/Menu/MainMenuButton.cs

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

354 lines
12 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.
2018-04-13 17:19:50 +08:00
2016-12-06 17:56:20 +08:00
using System;
using System.Linq;
using osu.Framework;
2016-12-05 20:09:41 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Audio.Sample;
2020-06-23 12:49:18 +08:00
using osu.Framework.Audio.Track;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Game.Graphics.Sprites;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
using osuTK.Input;
using osu.Framework.Extensions.Color4Extensions;
using osu.Game.Graphics.Containers;
2019-04-02 13:51:28 +08:00
using osu.Framework.Graphics.Effects;
using osu.Framework.Graphics.Sprites;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
2021-04-20 16:06:26 +08:00
using osu.Framework.Localisation;
using osu.Game.Beatmaps.ControlPoints;
2018-04-13 17:19:50 +08:00
2016-11-14 16:23:33 +08:00
namespace osu.Game.Screens.Menu
{
/// <summary>
/// Button designed specifically for the osu!next main menu.
/// In order to correctly flow, we have to use a negative margin on the parent container (due to the parallelogram shape).
/// </summary>
public partial class MainMenuButton : BeatSyncedContainer, IStateful<ButtonState>
{
public const float BOUNCE_COMPRESSION = 0.9f;
public const float HOVER_SCALE = 1.2f;
public const float BOUNCE_ROTATION = 8;
2023-11-24 10:05:57 +08:00
public event Action<ButtonState>? StateChanged;
2018-04-13 17:19:50 +08:00
public readonly Key[] TriggerKeys;
private readonly Container iconText;
private readonly Container box;
private readonly Box boxHoverLayer;
private readonly SpriteIcon icon;
private readonly string sampleName;
/// <summary>
/// The menu state for which we are visible for (assuming only one).
/// </summary>
public ButtonSystemState VisibleState
{
set
{
VisibleStateMin = value;
VisibleStateMax = value;
}
}
public ButtonSystemState VisibleStateMin = ButtonSystemState.TopLevel;
public ButtonSystemState VisibleStateMax = ButtonSystemState.TopLevel;
2023-11-24 10:05:57 +08:00
private readonly Action? clickAction;
private Sample? sampleClick;
private Sample? sampleHover;
private SampleChannel? sampleChannel;
2018-04-13 17:19:50 +08:00
public override bool IsPresent => base.IsPresent
// Allow keyboard interaction based on state rather than waiting for delayed animations.
|| state == ButtonState.Expanded;
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => box.ReceivePositionalInputAt(screenSpacePos);
2018-04-13 17:19:50 +08:00
2023-11-24 10:05:57 +08:00
public MainMenuButton(LocalisableString text, string sampleName, IconUsage symbol, Color4 colour, Action? clickAction = null, float extraWidth = 0, params Key[] triggerKeys)
{
this.sampleName = sampleName;
this.clickAction = clickAction;
TriggerKeys = triggerKeys;
2018-04-13 17:19:50 +08:00
2016-10-22 16:50:42 +08:00
AutoSizeAxes = Axes.Both;
Alpha = 0;
2018-04-13 17:19:50 +08:00
Vector2 boxSize = new Vector2(ButtonSystem.BUTTON_WIDTH + Math.Abs(extraWidth), ButtonArea.BUTTON_AREA_HEIGHT);
2018-04-13 17:19:50 +08:00
Children = new Drawable[]
{
2016-11-30 13:38:45 +08:00
box = new Container
{
2019-04-05 17:21:54 +08:00
// box needs to be always present to ensure the button is always sized correctly for flow
AlwaysPresent = true,
2016-11-30 13:38:45 +08:00
Masking = true,
MaskingSmoothness = 2,
2017-06-12 11:48:47 +08:00
EdgeEffect = new EdgeEffectParameters
2016-11-30 13:38:45 +08:00
{
Type = EdgeEffectType.Shadow,
2017-01-11 02:44:40 +08:00
Colour = Color4.Black.Opacity(0.2f),
2016-11-30 13:38:45 +08:00
Roundness = 5,
Radius = 8,
},
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Scale = new Vector2(0, 1),
Size = boxSize,
2017-02-07 15:15:45 +08:00
Shear = new Vector2(ButtonSystem.WEDGE_WIDTH / boxSize.Y, 0),
2017-06-07 19:53:37 +08:00
Children = new[]
2016-11-30 13:38:45 +08:00
{
new Box
{
EdgeSmoothness = new Vector2(1.5f, 0),
RelativeSizeAxes = Axes.Both,
Colour = colour,
},
boxHoverLayer = new Box
2016-11-30 13:38:45 +08:00
{
EdgeSmoothness = new Vector2(1.5f, 0),
2016-11-30 13:38:45 +08:00
RelativeSizeAxes = Axes.Both,
2019-08-21 12:29:50 +08:00
Blending = BlendingParameters.Additive,
Colour = Color4.White,
Alpha = 0,
2016-11-30 13:38:45 +08:00
},
}
},
iconText = new Container
{
AutoSizeAxes = Axes.Both,
Position = new Vector2(extraWidth / 2, 0),
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Children = new Drawable[]
{
icon = new SpriteIcon
{
Shadow = true,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
2023-12-29 07:15:23 +08:00
Size = new Vector2(32),
Position = new Vector2(0, 0),
Margin = new MarginPadding { Top = -4 },
Icon = symbol
},
new OsuSpriteText
{
Shadow = true,
AllowMultiline = false,
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Position = new Vector2(0, 35),
Margin = new MarginPadding { Left = -3 },
Text = text
}
}
}
};
}
2018-04-13 17:19:50 +08:00
private bool rightward;
2018-04-13 17:19:50 +08:00
2020-06-23 12:49:18 +08:00
protected override void OnNewBeat(int beatIndex, TimingControlPoint timingPoint, EffectControlPoint effectPoint, ChannelAmplitudes amplitudes)
{
base.OnNewBeat(beatIndex, timingPoint, effectPoint, amplitudes);
2018-04-13 17:19:50 +08:00
if (!IsHovered) return;
2018-04-13 17:19:50 +08:00
double duration = timingPoint.BeatLength / 2;
2018-04-13 17:19:50 +08:00
icon.RotateTo(rightward ? BOUNCE_ROTATION : -BOUNCE_ROTATION, duration * 2, Easing.InOutSine);
2018-04-13 17:19:50 +08:00
icon.Animate(
2017-07-23 02:50:25 +08:00
i => i.MoveToY(-10, duration, Easing.Out),
i => i.ScaleTo(HOVER_SCALE, duration, Easing.Out)
).Then(
2017-07-23 02:50:25 +08:00
i => i.MoveToY(0, duration, Easing.In),
i => i.ScaleTo(new Vector2(HOVER_SCALE, HOVER_SCALE * BOUNCE_COMPRESSION), duration, Easing.In)
);
2018-04-13 17:19:50 +08:00
rightward = !rightward;
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnHover(HoverEvent e)
{
if (State != ButtonState.Expanded) return true;
2018-04-13 17:19:50 +08:00
sampleHover?.Play();
2018-04-13 17:19:50 +08:00
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(1.5f, 1), 500, Easing.OutElastic);
2018-04-13 17:19:50 +08:00
double duration = TimeUntilNextBeat;
2018-04-13 17:19:50 +08:00
icon.ClearTransforms();
icon.RotateTo(rightward ? -BOUNCE_ROTATION : BOUNCE_ROTATION, duration, Easing.InOutSine);
icon.ScaleTo(new Vector2(HOVER_SCALE, HOVER_SCALE * BOUNCE_COMPRESSION), duration, Easing.Out);
return true;
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override void OnHoverLost(HoverLostEvent e)
{
2017-02-25 20:12:39 +08:00
icon.ClearTransforms();
2017-07-23 02:50:25 +08:00
icon.RotateTo(0, 500, Easing.Out);
icon.MoveTo(Vector2.Zero, 500, Easing.Out);
2017-07-23 02:50:25 +08:00
icon.ScaleTo(Vector2.One, 200, Easing.Out);
2018-04-13 17:19:50 +08:00
if (State == ButtonState.Expanded)
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(1, 1), 500, Easing.OutElastic);
}
2018-04-13 17:19:50 +08:00
2016-12-05 20:09:41 +08:00
[BackgroundDependencyLoader]
2018-08-31 06:04:40 +08:00
private void load(AudioManager audio)
2016-12-05 20:09:41 +08:00
{
sampleHover = audio.Samples.Get(@"Menu/button-hover");
sampleClick = audio.Samples.Get(!string.IsNullOrEmpty(sampleName) ? $@"Menu/{sampleName}" : @"UI/button-select");
2016-12-05 20:09:41 +08:00
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnMouseDown(MouseDownEvent e)
{
2017-07-23 02:50:25 +08:00
boxHoverLayer.FadeTo(0.1f, 1000, Easing.OutQuint);
2018-10-02 11:02:47 +08:00
return base.OnMouseDown(e);
}
2018-04-13 17:19:50 +08:00
protected override void OnMouseUp(MouseUpEvent e)
{
2017-07-23 02:50:25 +08:00
boxHoverLayer.FadeTo(0, 1000, Easing.OutQuint);
base.OnMouseUp(e);
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnClick(ClickEvent e)
{
trigger();
return true;
}
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnKeyDown(KeyDownEvent e)
{
if (e.Repeat || e.ControlPressed || e.ShiftPressed || e.AltPressed || e.SuperPressed)
return false;
2018-04-13 17:19:50 +08:00
if (TriggerKeys.Contains(e.Key))
{
trigger();
return true;
}
2018-04-13 17:19:50 +08:00
return false;
}
2018-04-13 17:19:50 +08:00
private void trigger()
{
sampleChannel = sampleClick?.GetChannel();
sampleChannel?.Play();
2018-04-13 17:19:50 +08:00
clickAction?.Invoke();
2018-04-13 17:19:50 +08:00
2017-02-25 20:12:39 +08:00
boxHoverLayer.ClearTransforms();
boxHoverLayer.Alpha = 0.9f;
2017-07-23 02:50:25 +08:00
boxHoverLayer.FadeOut(800, Easing.OutExpo);
}
2018-04-13 17:19:50 +08:00
public override bool HandleNonPositionalInput => state == ButtonState.Expanded;
public override bool HandlePositionalInput => state != ButtonState.Exploded && box.Scale.X >= 0.8f;
2018-04-13 17:19:50 +08:00
public void StopSamplePlayback() => sampleChannel?.Stop();
protected override void Update()
{
iconText.Alpha = Math.Clamp((box.Scale.X - 0.5f) / 0.3f, 0, 1);
base.Update();
}
2018-04-13 17:19:50 +08:00
public int ContractStyle;
2018-04-13 17:19:50 +08:00
2017-03-07 09:59:19 +08:00
private ButtonState state;
2018-04-13 17:19:50 +08:00
public ButtonState State
{
get => state;
2018-04-13 17:19:50 +08:00
2017-06-07 19:53:37 +08:00
set
{
if (state == value)
return;
2018-04-13 17:19:50 +08:00
state = value;
2018-04-13 17:19:50 +08:00
switch (state)
{
case ButtonState.Contracted:
switch (ContractStyle)
{
default:
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(0, 1), 500, Easing.OutExpo);
this.FadeOut(500);
break;
2019-04-01 11:44:46 +08:00
case 1:
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(0, 1), 400, Easing.InSine);
this.FadeOut(800);
break;
}
break;
2019-04-01 11:44:46 +08:00
case ButtonState.Expanded:
const int expand_duration = 500;
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(1, 1), expand_duration, Easing.OutExpo);
this.FadeIn(expand_duration / 6f);
break;
2019-04-01 11:44:46 +08:00
case ButtonState.Exploded:
const int explode_duration = 200;
2017-07-23 02:50:25 +08:00
box.ScaleTo(new Vector2(2, 1), explode_duration, Easing.OutExpo);
this.FadeOut(explode_duration / 4f * 3);
break;
}
2018-04-13 17:19:50 +08:00
2017-09-04 08:10:04 +08:00
StateChanged?.Invoke(State);
}
}
public ButtonSystemState ButtonSystemState
{
set
{
ContractStyle = 0;
switch (value)
{
case ButtonSystemState.Initial:
State = ButtonState.Contracted;
break;
2019-04-01 11:44:46 +08:00
case ButtonSystemState.EnteringMode:
ContractStyle = 1;
State = ButtonState.Contracted;
break;
2019-04-01 11:44:46 +08:00
default:
if (value <= VisibleStateMax && value >= VisibleStateMin)
State = ButtonState.Expanded;
else if (value < VisibleStateMin)
State = ButtonState.Contracted;
else
State = ButtonState.Exploded;
break;
}
}
}
}
2018-04-13 17:19:50 +08:00
public enum ButtonState
{
Contracted,
Expanded,
Exploded
}
}