1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-11 06:32:53 +08:00
osu-lazer/osu.Game/Screens/Menu/ButtonSystem.cs

461 lines
16 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;
using System.Linq;
using osu.Framework;
using osu.Framework.Allocation;
2018-04-30 01:15:09 +08:00
using osu.Framework.Audio;
using osu.Framework.Audio.Sample;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
2019-03-26 16:18:35 +08:00
using osu.Framework.Extensions.IEnumerableExtensions;
2023-11-24 09:56:59 +08:00
using osu.Framework.Extensions.LocalisationExtensions;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Input.Bindings;
using osu.Framework.Input.Events;
using osu.Framework.Logging;
using osu.Framework.Platform;
2018-04-30 01:15:09 +08:00
using osu.Framework.Threading;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
2019-03-26 16:18:35 +08:00
using osu.Game.Graphics.Containers;
using osu.Game.Input;
using osu.Game.Input.Bindings;
2021-04-20 16:06:26 +08:00
using osu.Game.Localisation;
using osu.Game.Online.API;
using osu.Game.Online.Rooms;
2018-06-06 15:17:51 +08:00
using osu.Game.Overlays;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
using osuTK.Input;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Menu
{
2022-11-24 13:32:20 +08:00
public partial class ButtonSystem : Container, IStateful<ButtonSystemState>, IKeyBindingHandler<GlobalAction>
2018-04-13 17:19:50 +08:00
{
2023-11-24 10:00:22 +08:00
public const float BUTTON_WIDTH = 140f;
public const float WEDGE_WIDTH = 20;
2018-04-13 17:19:50 +08:00
2023-11-24 10:00:22 +08:00
public event Action<ButtonSystemState>? StateChanged;
2023-11-24 10:00:22 +08:00
public Action? OnEditBeatmap;
public Action? OnEditSkin;
public Action<UIEvent>? OnExit;
2023-11-24 10:00:22 +08:00
public Action? OnBeatmapListing;
public Action? OnSolo;
public Action? OnSettings;
public Action? OnMultiplayer;
public Action? OnPlaylists;
public Action<Room>? OnDailyChallenge;
2018-04-13 17:19:50 +08:00
2023-11-24 10:00:22 +08:00
private readonly IBindable<bool> isIdle = new BindableBool();
2018-04-13 17:19:50 +08:00
2023-11-24 10:00:22 +08:00
private OsuLogo? logo;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Assign the <see cref="OsuLogo"/> that this ButtonSystem should manage the position of.
/// </summary>
/// <param name="logo">The instance of the logo to be assigned. If null, we are suspending from the screen that uses this ButtonSystem.</param>
2023-11-24 14:06:51 +08:00
public void SetOsuLogo(OsuLogo? logo)
2018-04-13 17:19:50 +08:00
{
this.logo = logo;
if (this.logo != null)
{
this.logo.Action = onOsuLogo;
// osuLogo.SizeForFlow relies on loading to be complete.
buttonArea.Flow.Position = new Vector2(WEDGE_WIDTH * 2 - (BUTTON_WIDTH + this.logo.SizeForFlow / 4), 0);
2018-04-13 17:19:50 +08:00
updateLogoState();
}
2019-03-26 16:18:35 +08:00
else
{
2019-03-27 10:37:16 +08:00
// We should stop tracking as the facade is now out of scope.
2019-04-08 14:24:09 +08:00
logoTrackingContainer.StopTracking();
2019-03-26 16:18:35 +08:00
}
2018-04-13 17:19:50 +08:00
}
private readonly ButtonArea buttonArea;
2018-04-13 17:19:50 +08:00
private readonly MainMenuButton backButton;
2018-04-13 17:19:50 +08:00
private readonly List<MainMenuButton> buttonsTopLevel = new List<MainMenuButton>();
private readonly List<MainMenuButton> buttonsPlay = new List<MainMenuButton>();
2023-11-24 09:56:59 +08:00
private readonly List<MainMenuButton> buttonsEdit = new List<MainMenuButton>();
2018-04-13 17:19:50 +08:00
2023-11-24 10:00:22 +08:00
private Sample? sampleBackToLogo;
private Sample? sampleLogoSwoosh;
2018-04-13 17:19:50 +08:00
private readonly LogoTrackingContainer logoTrackingContainer;
2019-03-26 16:18:35 +08:00
2022-04-19 14:46:03 +08:00
public bool ReturnToTopOnIdle { get; set; } = true;
2018-04-13 17:19:50 +08:00
public ButtonSystem()
{
RelativeSizeAxes = Axes.Both;
Child = logoTrackingContainer = new LogoTrackingContainer
2019-03-26 16:18:35 +08:00
{
RelativeSizeAxes = Axes.Both,
Child = buttonArea = new ButtonArea()
};
buttonArea.AddRange(new Drawable[]
2018-04-13 17:19:50 +08:00
{
new MainMenuButton(ButtonSystemStrings.Settings, string.Empty, OsuIcon.Settings, new Color4(85, 85, 85, 255), (_, _) => OnSettings?.Invoke(), Key.O, Key.S)
2023-11-24 09:56:59 +08:00
{
Padding = new MarginPadding { Right = WEDGE_WIDTH },
},
backButton = new MainMenuButton(ButtonSystemStrings.Back, @"back-to-top", OsuIcon.PrevCircle, new Color4(51, 58, 94, 255), (_, _) => State = ButtonSystemState.TopLevel)
{
Padding = new MarginPadding { Right = WEDGE_WIDTH },
VisibleStateMin = ButtonSystemState.Play,
VisibleStateMax = ButtonSystemState.Edit,
2018-04-13 17:19:50 +08:00
},
2019-04-17 16:24:09 +08:00
logoTrackingContainer.LogoFacade.With(d => d.Scale = new Vector2(0.74f))
});
buttonArea.Flow.CentreTarget = logoTrackingContainer.LogoFacade;
2018-04-13 17:19:50 +08:00
}
2023-11-24 10:00:22 +08:00
[Resolved]
private IAPIProvider api { get; set; } = null!;
[Resolved]
2023-11-24 10:00:22 +08:00
private OsuGame? game { get; set; }
2023-11-24 10:00:22 +08:00
[Resolved]
private LoginOverlay? loginOverlay { get; set; }
2023-11-24 10:00:22 +08:00
[BackgroundDependencyLoader]
private void load(AudioManager audio, IdleTracker? idleTracker, GameHost host)
2018-04-13 17:19:50 +08:00
{
buttonsPlay.Add(new MainMenuButton(ButtonSystemStrings.Solo, @"button-default-select", OsuIcon.Player, new Color4(102, 68, 204, 255), (_, _) => OnSolo?.Invoke(), Key.P)
{
Padding = new MarginPadding { Left = WEDGE_WIDTH },
});
buttonsPlay.Add(new MainMenuButton(ButtonSystemStrings.Multi, @"button-default-select", OsuIcon.Online, new Color4(94, 63, 186, 255), onMultiplayer, Key.M));
buttonsPlay.Add(new MainMenuButton(ButtonSystemStrings.Playlists, @"button-default-select", OsuIcon.Tournament, new Color4(94, 63, 186, 255), onPlaylists, Key.L));
buttonsPlay.Add(new DailyChallengeButton(@"button-daily-select", new Color4(94, 63, 186, 255), onDailyChallenge, Key.D));
buttonsPlay.ForEach(b => b.VisibleState = ButtonSystemState.Play);
buttonsEdit.Add(new MainMenuButton(EditorStrings.BeatmapEditor.ToLower(), @"button-default-select", OsuIcon.Beatmap, new Color4(238, 170, 0, 255), (_, _) => OnEditBeatmap?.Invoke(), Key.B, Key.E)
{
2024-05-17 17:50:23 +08:00
Padding = new MarginPadding { Left = WEDGE_WIDTH },
});
buttonsEdit.Add(new MainMenuButton(SkinEditorStrings.SkinEditor.ToLower(), @"button-default-select", OsuIcon.SkinB, new Color4(220, 160, 0, 255), (_, _) => OnEditSkin?.Invoke(), Key.S));
2023-11-24 09:56:59 +08:00
buttonsEdit.ForEach(b => b.VisibleState = ButtonSystemState.Edit);
buttonsTopLevel.Add(new MainMenuButton(ButtonSystemStrings.Play, @"button-play-select", OsuIcon.Logo, new Color4(102, 68, 204, 255), (_, _) => State = ButtonSystemState.Play, Key.P, Key.M, Key.L)
{
Padding = new MarginPadding { Left = WEDGE_WIDTH },
});
buttonsTopLevel.Add(new MainMenuButton(ButtonSystemStrings.Edit, @"button-play-select", OsuIcon.EditCircle, new Color4(238, 170, 0, 255), (_, _) => State = ButtonSystemState.Edit, Key.E));
buttonsTopLevel.Add(new MainMenuButton(ButtonSystemStrings.Browse, @"button-default-select", OsuIcon.Beatmap, new Color4(165, 204, 0, 255), (_, _) => OnBeatmapListing?.Invoke(), Key.B, Key.D));
if (host.CanExit)
buttonsTopLevel.Add(new MainMenuButton(ButtonSystemStrings.Exit, string.Empty, OsuIcon.CrossCircle, new Color4(238, 51, 153, 255), (_, e) => OnExit?.Invoke(e), Key.Q));
buttonArea.AddRange(buttonsPlay);
2023-11-24 09:56:59 +08:00
buttonArea.AddRange(buttonsEdit);
buttonArea.AddRange(buttonsTopLevel);
2019-03-26 16:18:35 +08:00
buttonArea.ForEach(b =>
{
if (b is MainMenuButton)
2019-03-26 16:18:35 +08:00
{
b.Origin = Anchor.CentreLeft;
b.Anchor = Anchor.CentreLeft;
}
});
isIdle.ValueChanged += idle => updateIdleState(idle.NewValue);
2018-11-26 15:50:41 +08:00
if (idleTracker != null) isIdle.BindTo(idleTracker.IsIdle);
2023-11-08 21:08:05 +08:00
sampleBackToLogo = audio.Samples.Get(@"Menu/back-to-logo");
sampleLogoSwoosh = audio.Samples.Get(@"Menu/osu-logo-swoosh");
2018-04-13 17:19:50 +08:00
}
private void onMultiplayer(MainMenuButton mainMenuButton, UIEvent uiEvent)
{
2020-12-29 17:56:29 +08:00
if (api.State.Value != APIState.Online)
{
loginOverlay?.Show();
return;
}
2018-12-26 12:18:36 +08:00
2020-12-22 13:55:25 +08:00
OnMultiplayer?.Invoke();
}
private void onPlaylists(MainMenuButton mainMenuButton, UIEvent uiEvent)
2020-12-22 13:55:25 +08:00
{
2020-12-29 17:56:29 +08:00
if (api.State.Value != APIState.Online)
2020-12-22 13:55:25 +08:00
{
loginOverlay?.Show();
2020-12-22 13:55:25 +08:00
return;
}
2020-12-25 12:11:21 +08:00
OnPlaylists?.Invoke();
}
private void onDailyChallenge(MainMenuButton button, UIEvent uiEvent)
{
if (api.State.Value != APIState.Online)
{
loginOverlay?.Show();
return;
}
var dailyChallengeButton = (DailyChallengeButton)button;
if (dailyChallengeButton.Room != null)
OnDailyChallenge?.Invoke(dailyChallengeButton.Room);
}
private void updateIdleState(bool isIdle)
{
2022-04-19 14:46:03 +08:00
if (!ReturnToTopOnIdle)
return;
if (isIdle && State != ButtonSystemState.Exit && State != ButtonSystemState.EnteringMode)
State = ButtonSystemState.Initial;
}
2022-07-05 17:15:37 +08:00
/// <summary>
/// Triggers the <see cref="logo"/> if the current <see cref="State"/> is <see cref="ButtonSystemState.Initial"/>.
/// </summary>
/// <returns><c>true</c> if the <see cref="logo"/> was triggered, <c>false</c> otherwise.</returns>
private bool triggerInitialOsuLogo()
{
if (State == ButtonSystemState.Initial)
{
StopSamplePlayback();
logo?.TriggerClick();
return true;
}
2022-07-05 17:15:37 +08:00
return false;
}
protected override bool OnKeyDown(KeyDownEvent e)
{
if (e.Repeat || e.ControlPressed || e.ShiftPressed || e.AltPressed || e.SuperPressed)
return false;
if (triggerInitialOsuLogo())
return true;
return base.OnKeyDown(e);
}
protected override bool OnJoystickPress(JoystickPressEvent e)
{
2022-07-05 17:15:37 +08:00
if (triggerInitialOsuLogo())
return true;
return base.OnJoystickPress(e);
}
protected override bool OnMidiDown(MidiDownEvent e)
{
2022-07-05 17:15:37 +08:00
if (triggerInitialOsuLogo())
return true;
return base.OnMidiDown(e);
}
2021-09-16 17:26:12 +08:00
public bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
2018-04-30 01:15:09 +08:00
{
if (e.Repeat)
return false;
2021-09-16 17:26:12 +08:00
switch (e.Action)
{
case GlobalAction.Back:
return goBack();
2019-04-01 11:16:05 +08:00
case GlobalAction.Select:
2021-08-04 16:27:44 +08:00
logo?.TriggerClick();
return true;
2019-04-01 11:16:05 +08:00
default:
return false;
}
}
2021-09-16 17:26:12 +08:00
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
{
}
private bool goBack()
{
switch (State)
{
case ButtonSystemState.TopLevel:
State = ButtonSystemState.Initial;
2023-11-08 21:08:05 +08:00
// Samples are explicitly played here in response to user interaction and not when transitioning due to idle.
StopSamplePlayback();
2023-11-08 21:08:05 +08:00
sampleBackToLogo?.Play();
return true;
2019-04-01 11:16:05 +08:00
2023-11-24 09:56:59 +08:00
case ButtonSystemState.Edit:
case ButtonSystemState.Play:
StopSamplePlayback();
2021-08-04 16:27:44 +08:00
backButton.TriggerClick();
return true;
2019-04-01 11:16:05 +08:00
default:
return false;
}
2018-04-30 01:15:09 +08:00
}
2018-04-13 17:19:50 +08:00
public void StopSamplePlayback()
{
buttonsPlay.ForEach(button => button.StopSamplePlayback());
buttonsTopLevel.ForEach(button => button.StopSamplePlayback());
logo?.StopSamplePlayback();
}
2018-04-13 17:19:50 +08:00
private bool onOsuLogo()
{
switch (state)
{
default:
2021-06-26 00:21:26 +08:00
return false;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.Initial:
State = ButtonSystemState.TopLevel;
2018-04-13 17:19:50 +08:00
return true;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.TopLevel:
2021-08-04 16:27:44 +08:00
buttonsTopLevel.First().TriggerClick();
2018-04-13 17:19:50 +08:00
return false;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.Play:
2021-08-04 16:27:44 +08:00
buttonsPlay.First().TriggerClick();
2018-04-13 17:19:50 +08:00
return false;
case ButtonSystemState.Edit:
buttonsEdit.First().TriggerClick();
return false;
2018-04-13 17:19:50 +08:00
}
}
private ButtonSystemState state = ButtonSystemState.Initial;
2018-04-13 17:19:50 +08:00
public override bool HandleNonPositionalInput => state != ButtonSystemState.Exit;
public override bool HandlePositionalInput => state != ButtonSystemState.Exit;
2018-04-13 17:19:50 +08:00
public ButtonSystemState State
2018-04-13 17:19:50 +08:00
{
get => state;
2018-04-13 17:19:50 +08:00
set
{
if (state == value) return;
ButtonSystemState lastState = state;
2018-04-13 17:19:50 +08:00
state = value;
updateLogoState(lastState);
Logger.Log($"{nameof(ButtonSystem)}'s state changed from {lastState} to {state}");
2018-04-13 17:19:50 +08:00
buttonArea.FinishTransforms(true);
2021-07-05 23:52:39 +08:00
using (buttonArea.BeginDelayedSequence(lastState == ButtonSystemState.Initial ? 150 : 0))
{
buttonArea.ButtonSystemState = state;
2018-04-13 17:19:50 +08:00
foreach (var b in buttonArea.OfType<MainMenuButton>())
b.ButtonSystemState = state;
2018-04-13 17:19:50 +08:00
}
StateChanged?.Invoke(State);
}
}
2023-11-24 10:00:22 +08:00
private ScheduledDelegate? logoDelayedAction;
2018-04-13 17:19:50 +08:00
private void updateLogoState(ButtonSystemState lastState = ButtonSystemState.Initial)
2018-04-13 17:19:50 +08:00
{
if (logo == null) return;
switch (state)
{
case ButtonSystemState.Exit:
case ButtonSystemState.Initial:
logoDelayedAction?.Cancel();
2018-04-13 17:19:50 +08:00
logoDelayedAction = Scheduler.AddDelayed(() =>
{
logoTrackingContainer.StopTracking();
game?.Toolbar.Hide();
2018-06-06 15:17:51 +08:00
logo?.ClearTransforms(targetMember: nameof(Position));
logo?.MoveTo(new Vector2(0.5f), 800, Easing.OutExpo);
logo?.ScaleTo(1, 800, Easing.OutExpo);
}, buttonArea.Alpha * 150);
if (lastState == ButtonSystemState.TopLevel)
sampleLogoSwoosh?.Play();
2018-04-13 17:19:50 +08:00
break;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.TopLevel:
case ButtonSystemState.Play:
2018-04-13 17:19:50 +08:00
switch (lastState)
{
case ButtonSystemState.TopLevel: // coming from toplevel to play
break;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.Initial:
logo.ClearTransforms(targetMember: nameof(Position));
bool impact = logo.Scale.X > 0.6f;
logo.ScaleTo(0.5f, 200, Easing.In);
2018-04-13 17:19:50 +08:00
logoTrackingContainer.StartTracking(logo, 200, Easing.In);
2018-04-13 17:19:50 +08:00
logoDelayedAction?.Cancel();
2018-04-13 17:19:50 +08:00
logoDelayedAction = Scheduler.AddDelayed(() =>
{
2018-05-31 14:17:59 +08:00
if (impact)
logo?.Impact();
2018-06-06 15:17:51 +08:00
game?.Toolbar.Show();
2018-04-13 17:19:50 +08:00
}, 200);
break;
2019-04-01 11:16:05 +08:00
2018-04-13 17:19:50 +08:00
default:
logo.ClearTransforms(targetMember: nameof(Position));
2019-04-08 14:24:09 +08:00
logoTrackingContainer.StartTracking(logo, 0, Easing.In);
2018-04-13 17:19:50 +08:00
logo.ScaleTo(0.5f, 200, Easing.OutQuint);
break;
}
2018-04-30 01:15:09 +08:00
2018-04-13 17:19:50 +08:00
break;
2019-04-01 11:16:05 +08:00
case ButtonSystemState.EnteringMode:
2019-07-05 15:07:17 +08:00
logoTrackingContainer.StartTracking(logo, lastState == ButtonSystemState.Initial ? MainMenu.FADE_OUT_DURATION : 0, Easing.InSine);
2018-04-13 17:19:50 +08:00
break;
}
}
}
public enum ButtonSystemState
2018-04-13 17:19:50 +08:00
{
Exit,
2018-04-13 17:19:50 +08:00
Initial,
TopLevel,
Play,
2023-11-24 09:56:59 +08:00
Edit,
2018-04-13 17:19:50 +08:00
EnteringMode,
}
}