1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-08 06:52:59 +08:00
osu-lazer/osu.Game/Screens/Play/GameplayMenuOverlay.cs

331 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.
2018-04-13 17:19:50 +08:00
using System;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics.Sprites;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
using osu.Framework.Allocation;
using osu.Game.Graphics.UserInterface;
using osu.Framework.Graphics.Shapes;
2018-11-20 15:51:59 +08:00
using osuTK.Input;
2018-04-13 17:19:50 +08:00
using System.Collections.Generic;
using System.Linq;
using osu.Framework.Input.Bindings;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
using osu.Game.Input.Bindings;
using Humanizer;
2019-04-02 13:51:28 +08:00
using osu.Framework.Graphics.Effects;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Play
{
public abstract class GameplayMenuOverlay : OverlayContainer, IKeyBindingHandler<GlobalAction>
2018-04-13 17:19:50 +08:00
{
private const int transition_duration = 200;
private const int button_height = 70;
private const float background_alpha = 0.75f;
protected override bool BlockNonPositionalInput => true;
2018-04-13 17:19:50 +08:00
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => true;
2018-04-13 17:19:50 +08:00
public Action OnRetry;
public Action OnQuit;
/// <summary>
/// Action that is invoked when <see cref="GlobalAction.Back"/> is triggered.
/// </summary>
2019-03-24 11:03:06 +08:00
protected virtual Action BackAction => () => InternalButtons.Children.LastOrDefault()?.Click();
/// <summary>
/// Action that is invoked when <see cref="GlobalAction.Select"/> is triggered.
/// </summary>
2019-03-24 11:03:06 +08:00
protected virtual Action SelectAction => () => InternalButtons.Children.FirstOrDefault(f => f.Selected.Value)?.Click();
2018-04-13 17:19:50 +08:00
public abstract string Header { get; }
2019-03-24 11:03:06 +08:00
2018-04-13 17:19:50 +08:00
public abstract string Description { get; }
protected internal FillFlowContainer<DialogButton> InternalButtons;
public IReadOnlyList<DialogButton> Buttons => InternalButtons;
private FillFlowContainer retryCounterContainer;
protected GameplayMenuOverlay()
{
RelativeSizeAxes = Axes.Both;
State.ValueChanged += s => selectionIndex = -1;
2018-04-13 17:19:50 +08:00
}
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
Children = new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = Color4.Black,
Alpha = background_alpha,
},
new FillFlowContainer
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(0, 50),
Origin = Anchor.Centre,
Anchor = Anchor.Centre,
Children = new Drawable[]
{
new FillFlowContainer
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(0, 20),
Children = new Drawable[]
{
new OsuSpriteText
{
Text = Header,
2019-02-20 15:52:36 +08:00
Font = OsuFont.GetFont(size: 30),
2018-04-13 17:19:50 +08:00
Spacing = new Vector2(5, 0),
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Colour = colours.Yellow,
Shadow = true,
ShadowColour = new Color4(0, 0, 0, 0.25f)
},
new OsuSpriteText
{
Text = Description,
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Shadow = true,
ShadowColour = new Color4(0, 0, 0, 0.25f)
}
}
},
InternalButtons = new FillFlowContainer<DialogButton>
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Masking = true,
EdgeEffect = new EdgeEffectParameters
{
Type = EdgeEffectType.Shadow,
Colour = Color4.Black.Opacity(0.6f),
Radius = 50
},
},
retryCounterContainer = new FillFlowContainer
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
AutoSizeAxes = Axes.Both,
}
}
},
};
updateRetryCount();
}
private int retries;
public int Retries
{
set
{
if (value == retries)
return;
retries = value;
if (retryCounterContainer != null)
updateRetryCount();
}
}
protected override void PopIn() => this.FadeIn(transition_duration, Easing.In);
protected override void PopOut() => this.FadeOut(transition_duration, Easing.In);
// Don't let mouse down events through the overlay or people can click circles while paused.
2018-10-02 11:02:47 +08:00
protected override bool OnMouseDown(MouseDownEvent e) => true;
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnMouseUp(MouseUpEvent e) => true;
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnMouseMove(MouseMoveEvent e) => true;
2018-04-13 17:19:50 +08:00
protected void AddButton(string text, Color4 colour, Action action)
{
var button = new Button
{
Text = text,
ButtonColour = colour,
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Height = button_height,
Action = delegate
{
action?.Invoke();
Hide();
}
};
button.Selected.ValueChanged += selected => buttonSelectionChanged(button, selected.NewValue);
2018-04-13 17:19:50 +08:00
InternalButtons.Add(button);
}
2019-12-11 13:10:35 +08:00
private int selectionIndex = -1;
2018-04-13 17:19:50 +08:00
2019-12-11 13:10:35 +08:00
private void setSelected(int value)
2018-04-13 17:19:50 +08:00
{
2019-12-11 13:10:35 +08:00
if (selectionIndex == value)
return;
2018-04-13 17:19:50 +08:00
2019-12-11 13:10:35 +08:00
// Deselect the previously-selected button
if (selectionIndex != -1)
InternalButtons[selectionIndex].Selected.Value = false;
2018-04-13 17:19:50 +08:00
2019-12-11 13:10:35 +08:00
selectionIndex = value;
2018-04-13 17:19:50 +08:00
2019-12-11 13:10:35 +08:00
// Select the newly-selected button
if (selectionIndex != -1)
InternalButtons[selectionIndex].Selected.Value = true;
2018-04-13 17:19:50 +08:00
}
2018-10-02 11:02:47 +08:00
protected override bool OnKeyDown(KeyDownEvent e)
2018-04-13 17:19:50 +08:00
{
2018-10-02 11:02:47 +08:00
if (!e.Repeat)
2018-04-13 17:19:50 +08:00
{
2018-10-02 11:02:47 +08:00
switch (e.Key)
2018-04-13 17:19:50 +08:00
{
case Key.Up:
if (selectionIndex == -1 || selectionIndex == 0)
2019-12-11 13:10:35 +08:00
setSelected(InternalButtons.Count - 1);
2018-04-13 17:19:50 +08:00
else
2019-12-11 13:34:17 +08:00
setSelected(selectionIndex - 1);
2018-04-13 17:19:50 +08:00
return true;
2019-04-01 11:44:46 +08:00
2018-04-13 17:19:50 +08:00
case Key.Down:
if (selectionIndex == -1 || selectionIndex == InternalButtons.Count - 1)
2019-12-11 13:10:35 +08:00
setSelected(0);
2018-04-13 17:19:50 +08:00
else
2019-12-11 13:34:17 +08:00
setSelected(selectionIndex + 1);
2018-04-13 17:19:50 +08:00
return true;
}
}
2018-10-02 11:02:47 +08:00
return base.OnKeyDown(e);
2018-04-13 17:19:50 +08:00
}
public bool OnPressed(GlobalAction action)
{
2019-03-24 11:03:06 +08:00
switch (action)
{
2019-03-24 11:03:06 +08:00
case GlobalAction.Back:
BackAction.Invoke();
return true;
2019-04-01 11:44:46 +08:00
2019-03-24 11:03:06 +08:00
case GlobalAction.Select:
SelectAction.Invoke();
return true;
}
2019-03-24 11:19:09 +08:00
return false;
}
2019-03-24 11:19:09 +08:00
public bool OnReleased(GlobalAction action)
{
switch (action)
{
case GlobalAction.Back:
case GlobalAction.Select:
return true;
}
return false;
}
2018-04-13 17:19:50 +08:00
private void buttonSelectionChanged(DialogButton button, bool isSelected)
{
if (!isSelected)
2019-12-11 13:34:17 +08:00
setSelected(-1);
2018-04-13 17:19:50 +08:00
else
2019-12-11 13:34:17 +08:00
setSelected(InternalButtons.IndexOf(button));
2018-04-13 17:19:50 +08:00
}
private void updateRetryCount()
{
// "You've retried 1,065 times in this session"
// "You've retried 1 time in this session"
retryCounterContainer.Children = new Drawable[]
{
new OsuSpriteText
{
Text = "You've retried ",
Shadow = true,
ShadowColour = new Color4(0, 0, 0, 0.25f),
Font = OsuFont.GetFont(size: 18),
2018-04-13 17:19:50 +08:00
},
new OsuSpriteText
{
Text = "time".ToQuantity(retries),
Font = OsuFont.GetFont(weight: FontWeight.Bold, size: 18),
2018-04-13 17:19:50 +08:00
Shadow = true,
ShadowColour = new Color4(0, 0, 0, 0.25f),
},
new OsuSpriteText
{
Text = " in this session",
2018-04-13 17:19:50 +08:00
Shadow = true,
ShadowColour = new Color4(0, 0, 0, 0.25f),
Font = OsuFont.GetFont(size: 18),
2018-04-13 17:19:50 +08:00
}
};
}
private class Button : DialogButton
{
// required to ensure keyboard navigation always starts from an extremity (unless the cursor is moved)
protected override bool OnHover(HoverEvent e) => true;
2018-10-02 11:02:47 +08:00
protected override bool OnMouseMove(MouseMoveEvent e)
2018-04-13 17:19:50 +08:00
{
Selected.Value = true;
2018-10-02 11:02:47 +08:00
return base.OnMouseMove(e);
2018-04-13 17:19:50 +08:00
}
}
[Resolved]
private GlobalActionContainer globalAction { get; set; }
protected override bool Handle(UIEvent e)
{
switch (e)
{
case ScrollEvent _:
if (ReceivePositionalInputAt(e.ScreenSpaceMousePosition))
return globalAction.TriggerEvent(e);
break;
}
return base.Handle(e);
}
2018-04-13 17:19:50 +08:00
}
}