1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-10 20:23:51 +08:00
osu-lazer/osu.Game/Screens/Play/GameplayMenuOverlay.cs

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

283 lines
9.4 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 System.Collections.Generic;
2021-07-07 18:59:31 +08:00
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Effects;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Input.Bindings;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
2023-05-27 18:29:14 +08:00
using osu.Framework.Localisation;
using osu.Game.Beatmaps;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.Sprites;
using osu.Game.Graphics.UserInterface;
using osu.Game.Input.Bindings;
using osuTK;
using osuTK.Graphics;
using osu.Game.Localisation;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Play
{
public abstract partial class GameplayMenuOverlay : OverlayContainer, IKeyBindingHandler<GlobalAction>
{
protected const int TRANSITION_DURATION = 200;
private const int button_height = 70;
private const float background_alpha = 0.75f;
2018-04-13 17:19:50 +08:00
protected override bool BlockScrollInput => false;
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos) => true;
2018-04-13 17:19:50 +08:00
public Action? OnResume { get; init; }
public Action? OnRetry { get; init; }
public Action? OnQuit { get; init; }
2018-04-13 17:19:50 +08:00
/// <summary>
/// Action that is invoked when <see cref="GlobalAction.Back"/> is triggered.
/// </summary>
protected virtual Action BackAction => () =>
{
// We prefer triggering the button click as it will animate...
// but sometimes buttons aren't present (see FailOverlay's constructor as an example).
if (Buttons.Any())
Buttons.Last().TriggerClick();
else
OnQuit?.Invoke();
};
/// <summary>
/// Action that is invoked when <see cref="GlobalAction.Select"/> is triggered.
/// </summary>
2021-08-04 16:27:44 +08:00
protected virtual Action SelectAction => () => InternalButtons.Selected?.TriggerClick();
2019-03-24 11:03:06 +08:00
2023-05-27 18:29:14 +08:00
public abstract LocalisableString Header { get; }
2019-03-24 11:03:06 +08:00
protected SelectionCycleFillFlowContainer<DialogButton> InternalButtons = null!;
2017-12-18 18:13:08 +08:00
public IReadOnlyList<DialogButton> Buttons => InternalButtons;
2018-04-13 17:19:50 +08:00
private TextFlowContainer playInfoText = null!;
[Resolved]
private GlobalActionContainer globalAction { get; set; } = null!;
2018-04-13 17:19:50 +08:00
2017-12-18 15:40:50 +08:00
protected GameplayMenuOverlay()
{
2017-12-18 13:05:12 +08:00
RelativeSizeAxes = Axes.Both;
}
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 OsuSpriteText
{
Text = Header,
Font = OsuFont.GetFont(size: 48),
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Colour = colours.Yellow,
},
InternalButtons = new SelectionCycleFillFlowContainer<DialogButton>
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Masking = true,
2017-06-12 11:48:47 +08:00
EdgeEffect = new EdgeEffectParameters
{
Type = EdgeEffectType.Shadow,
Colour = Color4.Black.Opacity(0.6f),
Radius = 50
},
},
playInfoText = new OsuTextFlowContainer(cp => cp.Font = OsuFont.GetFont(size: 18))
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
TextAnchor = Anchor.TopCentre,
AutoSizeAxes = Axes.Both,
}
}
},
};
2018-04-13 17:19:50 +08:00
if (OnResume != null)
AddButton(GameplayMenuOverlayStrings.Continue, colours.Green, () => OnResume.Invoke());
if (OnRetry != null)
AddButton(GameplayMenuOverlayStrings.Retry, colours.YellowDark, () => OnRetry.Invoke());
if (OnQuit != null)
AddButton(GameplayMenuOverlayStrings.Quit, new Color4(170, 27, 39, 255), () => OnQuit.Invoke());
2022-06-24 20:25:23 +08:00
State.ValueChanged += _ => InternalButtons.Deselect();
updateInfoText();
}
2018-04-13 17:19:50 +08:00
private int retries;
2018-04-13 17:19:50 +08:00
2017-12-18 13:05:12 +08:00
public int Retries
{
2017-12-18 13:05:12 +08:00
set
{
if (value == retries)
return;
2018-04-13 17:19:50 +08:00
retries = value;
if (IsLoaded)
updateInfoText();
2017-12-18 13:05:12 +08:00
}
}
2018-04-13 17:19:50 +08:00
protected override void PopIn()
{
this.FadeIn(TRANSITION_DURATION, Easing.In);
updateInfoText();
}
protected override void PopOut() => this.FadeOut(TRANSITION_DURATION, Easing.In);
2018-04-13 17:19:50 +08:00
2017-12-18 13:05:12 +08:00
// 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 OnMouseMove(MouseMoveEvent e) => true;
2018-04-13 17:19:50 +08:00
2023-05-27 18:29:14 +08:00
protected void AddButton(LocalisableString text, Color4 colour, Action? action)
2017-12-18 13:05:12 +08:00
{
2017-12-18 15:40:50 +08:00
var button = new Button
2017-12-18 13:05:12 +08:00
{
Text = text,
ButtonColour = colour,
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Height = button_height,
Action = delegate
{
action?.Invoke();
Hide();
}
};
2018-04-13 17:19:50 +08:00
2017-12-18 18:13:08 +08:00
InternalButtons.Add(button);
}
2018-04-13 17:19:50 +08:00
public virtual bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
{
2021-09-16 17:26:12 +08:00
switch (e.Action)
{
case GlobalAction.SelectPrevious:
InternalButtons.SelectPrevious();
return true;
case GlobalAction.SelectNext:
InternalButtons.SelectNext();
return true;
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;
}
2021-09-16 17:26:12 +08:00
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
2019-03-24 11:19:09 +08:00
{
}
2021-07-06 20:11:46 +08:00
[Resolved]
private IGameplayClock? gameplayClock { get; set; }
[Resolved]
private GameplayState? gameplayState { get; set; }
2018-04-13 17:19:50 +08:00
private void updateInfoText()
{
playInfoText.Clear();
playInfoText.AddText(GameplayMenuOverlayStrings.RetryCount);
playInfoText.AddText(retries.ToString(), cp => cp.Font = cp.Font.With(weight: FontWeight.Bold));
if (getSongProgress() is int progress)
{
playInfoText.NewLine();
playInfoText.AddText(GameplayMenuOverlayStrings.SongProgress);
playInfoText.AddText($"{progress}%", cp => cp.Font = cp.Font.With(weight: FontWeight.Bold));
}
}
2018-04-13 17:19:50 +08:00
private int? getSongProgress()
{
if (gameplayClock == null || gameplayState == null)
return null;
(double firstHitTime, double lastHitTime) = gameplayState.Beatmap.CalculatePlayableBounds();
double playableLength = (lastHitTime - firstHitTime);
if (playableLength == 0)
return 0;
return (int)Math.Clamp(((gameplayClock.CurrentTime - firstHitTime) / playableLength) * 100, 0, 100);
}
2017-12-18 15:40:50 +08:00
private partial 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)
{
State = SelectionState.Selected;
2018-10-02 11:02:47 +08:00
return base.OnMouseMove(e);
}
}
protected override bool Handle(UIEvent e)
{
switch (e)
{
2022-06-24 20:25:23 +08:00
case ScrollEvent:
if (ReceivePositionalInputAt(e.ScreenSpaceMousePosition))
return globalAction.TriggerEvent(e);
break;
}
return base.Handle(e);
}
}
}