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

471 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
2022-06-17 15:37:17 +08:00
#nullable disable
using System;
using System.Diagnostics;
using System.Linq;
using JetBrains.Annotations;
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Audio.Sample;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
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-13 17:19:50 +08:00
using osu.Framework.Screens;
using osu.Game.Beatmaps;
using osu.Game.Configuration;
using osu.Game.Graphics;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics.Containers;
using osu.Game.Input.Bindings;
2020-07-06 21:03:09 +08:00
using osu.Game.IO;
2019-09-08 13:36:58 +08:00
using osu.Game.Online.API;
using osu.Game.Overlays;
using osu.Game.Overlays.Dialog;
2023-11-24 09:56:59 +08:00
using osu.Game.Overlays.SkinEditor;
using osu.Game.Rulesets;
2018-04-13 17:19:50 +08:00
using osu.Game.Screens.Backgrounds;
using osu.Game.Screens.Edit;
using osu.Game.Screens.OnlinePlay.DailyChallenge;
using osu.Game.Screens.OnlinePlay.Multiplayer;
using osu.Game.Screens.OnlinePlay.Playlists;
2018-04-13 17:19:50 +08:00
using osu.Game.Screens.Select;
2021-05-19 15:27:38 +08:00
using osuTK;
using osuTK.Graphics;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Menu
{
2022-11-24 13:32:20 +08:00
public partial class MainMenu : OsuScreen, IHandlePresentBeatmap, IKeyBindingHandler<GlobalAction>
2018-04-13 17:19:50 +08:00
{
public const float FADE_IN_DURATION = 300;
public const float FADE_OUT_DURATION = 400;
2018-04-13 17:19:50 +08:00
2022-04-19 14:46:03 +08:00
public override bool HideOverlaysOnEnter => Buttons == null || Buttons.State == ButtonSystemState.Initial;
2018-04-13 17:19:50 +08:00
2019-06-25 17:38:14 +08:00
public override bool AllowBackButton => false;
public override bool AllowExternalScreenChange => true;
public override bool? AllowGlobalTrackControl => true;
2018-04-13 17:19:50 +08:00
private Screen songSelect;
private MenuSideFlashes sideFlashes;
2018-04-13 17:19:50 +08:00
2022-04-19 14:46:03 +08:00
protected ButtonSystem Buttons;
[Resolved]
private GameHost host { get; set; }
2018-04-13 17:19:50 +08:00
[Resolved]
private INotificationOverlay notifications { get; set; }
2020-08-11 11:40:58 +08:00
[Resolved]
2020-08-16 22:04:49 +08:00
private MusicController musicController { get; set; }
2019-09-08 13:36:58 +08:00
[Resolved]
private IAPIProvider api { get; set; }
[Resolved]
private Storage storage { get; set; }
[Resolved(canBeNull: true)]
private LoginOverlay login { get; set; }
2019-09-19 16:40:46 +08:00
[Resolved(canBeNull: true)]
private IDialogOverlay dialogOverlay { get; set; }
[Resolved(canBeNull: true)]
private VersionManager versionManager { get; set; }
protected override BackgroundScreen CreateBackground() => new BackgroundScreenDefault();
2018-04-13 17:19:50 +08:00
protected override bool PlayExitSound => false;
private Bindable<double> holdDelay;
2019-09-25 20:56:47 +08:00
private Bindable<bool> loginDisplayed;
private HoldToExitGameOverlay holdToExitGameOverlay;
2019-09-19 19:17:58 +08:00
private bool exitConfirmedViaDialog;
private bool exitConfirmedViaHoldOrClick;
private ParallaxContainer buttonsContainer;
2020-01-11 12:17:13 +08:00
private SongTicker songTicker;
private Container logoTarget;
private OnlineMenuBanner onlineMenuBanner;
private MenuTip menuTip;
private FillFlowContainer bottomElementsFlow;
2023-12-28 16:14:16 +08:00
private SupporterDisplay supporterDisplay;
private Sample reappearSampleSwoosh;
2023-11-24 09:56:59 +08:00
[Resolved(canBeNull: true)]
private SkinEditorOverlay skinEditor { get; set; }
[BackgroundDependencyLoader(true)]
private void load(BeatmapListingOverlay beatmapListing, SettingsOverlay settings, OsuConfigManager config, SessionStatics statics, AudioManager audio)
2018-04-13 17:19:50 +08:00
{
holdDelay = config.GetBindable<double>(OsuSetting.UIHoldActivationDelay);
2019-09-28 20:21:51 +08:00
loginDisplayed = statics.GetBindable<bool>(Static.LoginOverlayDisplayed);
2020-01-24 15:34:13 +08:00
if (host.CanExit)
{
AddInternal(holdToExitGameOverlay = new HoldToExitGameOverlay
2020-01-24 15:34:13 +08:00
{
Action = () =>
{
exitConfirmedViaHoldOrClick = holdDelay.Value > 0;
this.Exit();
2020-01-24 15:34:13 +08:00
}
});
}
AddRangeInternal(new[]
2018-04-13 17:19:50 +08:00
{
buttonsContainer = new ParallaxContainer
2018-04-13 17:19:50 +08:00
{
ParallaxAmount = 0.01f,
Children = new Drawable[]
{
2022-04-19 14:46:03 +08:00
Buttons = new ButtonSystem
2018-04-13 17:19:50 +08:00
{
2023-11-24 09:56:59 +08:00
OnEditBeatmap = () =>
{
Beatmap.SetDefault();
this.Push(new EditorLoader());
},
2023-11-24 09:56:59 +08:00
OnEditSkin = () =>
{
skinEditor?.Show();
},
OnSolo = loadSoloSongSelect,
2020-12-25 12:38:11 +08:00
OnMultiplayer = () => this.Push(new Multiplayer()),
2020-12-25 22:45:44 +08:00
OnPlaylists = () => this.Push(new Playlists()),
OnDailyChallenge = room =>
{
this.Push(new DailyChallenge(room));
},
OnExit = () =>
{
exitConfirmedViaHoldOrClick = true;
this.Exit();
}
2018-04-13 17:19:50 +08:00
}
}
},
logoTarget = new Container { RelativeSizeAxes = Axes.Both, },
2018-04-13 17:19:50 +08:00
sideFlashes = new MenuSideFlashes(),
songTicker = new SongTicker
{
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
Margin = new MarginPadding { Right = 15, Top = 5 }
2020-01-24 15:34:13 +08:00
},
2023-07-13 19:31:00 +08:00
new KiaiMenuFountains(),
bottomElementsFlow = new FillFlowContainer
{
AutoSizeAxes = Axes.Both,
Direction = FillDirection.Vertical,
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
2023-12-28 15:16:27 +08:00
Spacing = new Vector2(5),
Children = new Drawable[]
{
menuTip = new MenuTip
{
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
},
onlineMenuBanner = new OnlineMenuBanner
{
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
}
}
},
2023-12-28 16:14:16 +08:00
supporterDisplay = new SupporterDisplay
{
Margin = new MarginPadding(5),
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
},
holdToExitGameOverlay?.CreateProxy() ?? Empty()
});
2022-04-19 14:46:03 +08:00
Buttons.StateChanged += state =>
{
switch (state)
{
case ButtonSystemState.Initial:
case ButtonSystemState.Exit:
ApplyToBackground(b => b.FadeColour(Color4.White, 500, Easing.OutSine));
onlineMenuBanner.State.Value = Visibility.Hidden;
break;
2019-04-01 11:44:46 +08:00
default:
ApplyToBackground(b => b.FadeColour(OsuColour.Gray(0.8f), 500, Easing.OutSine));
onlineMenuBanner.State.Value = Visibility.Visible;
break;
}
};
2018-04-13 17:19:50 +08:00
2022-04-19 14:46:03 +08:00
Buttons.OnSettings = () => settings?.ToggleVisibility();
Buttons.OnBeatmapListing = () => beatmapListing?.ToggleVisibility();
2018-04-13 17:19:50 +08:00
reappearSampleSwoosh = audio.Samples.Get(@"Menu/reappear-swoosh");
2018-04-13 17:19:50 +08:00
preloadSongSelect();
}
public void ReturnToOsuLogo() => Buttons.State = ButtonSystemState.Initial;
2018-04-13 17:19:50 +08:00
private void preloadSongSelect()
{
if (songSelect == null)
LoadComponentAsync(songSelect = new PlaySongSelect());
}
private void loadSoloSongSelect() => this.Push(consumeSongSelect());
2018-04-13 17:19:50 +08:00
private Screen consumeSongSelect()
{
var s = songSelect;
songSelect = null;
return s;
}
public override void OnEntering(ScreenTransitionEvent e)
2018-04-13 17:19:50 +08:00
{
base.OnEntering(e);
2022-04-19 14:46:03 +08:00
Buttons.FadeInFromZero(500);
2018-04-13 17:19:50 +08:00
if (e.Last is IntroScreen && musicController.TrackLoaded)
2018-04-13 17:19:50 +08:00
{
var track = musicController.CurrentTrack;
// presume the track is the current beatmap's track. not sure how correct this assumption is but it has worked until now.
if (!track.IsRunning)
2018-04-13 17:19:50 +08:00
{
Beatmap.Value.PrepareTrackForPreview(false);
track.Restart();
2018-04-13 17:19:50 +08:00
}
}
2020-07-06 21:03:09 +08:00
if (storage is OsuStorage osuStorage && osuStorage.Error != OsuStorageError.None)
dialogOverlay?.Push(new StorageErrorDialog(osuStorage, osuStorage.Error));
2018-04-13 17:19:50 +08:00
}
[CanBeNull]
private Drawable proxiedLogo;
2018-04-13 17:19:50 +08:00
protected override void LogoArriving(OsuLogo logo, bool resuming)
{
base.LogoArriving(logo, resuming);
2022-04-19 14:46:03 +08:00
Buttons.SetOsuLogo(logo);
2018-04-13 17:19:50 +08:00
logo.FadeColour(Color4.White, 100, Easing.OutQuint);
logo.FadeIn(100, Easing.OutQuint);
proxiedLogo = logo.ProxyToContainer(logoTarget);
2018-04-13 17:19:50 +08:00
if (resuming)
{
2022-04-19 14:46:03 +08:00
Buttons.State = ButtonSystemState.TopLevel;
2018-04-13 17:19:50 +08:00
this.FadeIn(FADE_IN_DURATION, Easing.OutQuint);
buttonsContainer.MoveTo(new Vector2(0, 0), FADE_IN_DURATION, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
sideFlashes.Delay(FADE_IN_DURATION).FadeIn(64, Easing.InQuint);
2018-04-13 17:19:50 +08:00
}
else if (!api.IsLoggedIn || api.State.Value == APIState.RequiresSecondFactorAuth)
{
// copy out old action to avoid accidentally capturing logo.Action in closure, causing a self-reference loop.
var previousAction = logo.Action;
// we want to hook into logo.Action to display the login overlay, but also preserve the return value of the old action.
// therefore pass the old action to displayLogin, so that it can return that value.
// this ensures that the OsuLogo sample does not play when it is not desired.
logo.Action = () => displayLogin(previousAction);
}
bool displayLogin(Func<bool> originalAction)
{
2019-09-25 20:56:47 +08:00
if (!loginDisplayed.Value)
{
Scheduler.AddDelayed(() => login?.Show(), 500);
2019-09-25 20:56:47 +08:00
loginDisplayed.Value = true;
}
return originalAction.Invoke();
}
2018-04-13 17:19:50 +08:00
}
protected override void Update()
{
base.Update();
bottomElementsFlow.Margin = new MarginPadding
{
Bottom = (versionManager?.DrawHeight + 5) ?? 0
};
}
2018-04-13 17:19:50 +08:00
protected override void LogoSuspending(OsuLogo logo)
{
var seq = logo.FadeOut(300, Easing.InSine)
.ScaleTo(0.2f, 300, Easing.InSine);
if (proxiedLogo != null)
{
logo.ReturnProxy();
proxiedLogo = null;
}
2022-04-19 14:46:03 +08:00
seq.OnComplete(_ => Buttons.SetOsuLogo(null));
seq.OnAbort(_ => Buttons.SetOsuLogo(null));
2018-04-13 17:19:50 +08:00
}
protected override void LogoExiting(OsuLogo logo)
{
base.LogoExiting(logo);
if (proxiedLogo != null)
{
logo.ReturnProxy();
proxiedLogo = null;
}
}
public override void OnSuspending(ScreenTransitionEvent e)
2018-04-13 17:19:50 +08:00
{
base.OnSuspending(e);
2018-04-13 17:19:50 +08:00
2022-04-19 14:46:03 +08:00
Buttons.State = ButtonSystemState.EnteringMode;
2018-04-13 17:19:50 +08:00
this.FadeOut(FADE_OUT_DURATION, Easing.InSine);
buttonsContainer.MoveTo(new Vector2(-800, 0), FADE_OUT_DURATION, Easing.InSine);
2018-04-13 17:19:50 +08:00
sideFlashes.FadeOut(64, Easing.OutQuint);
bottomElementsFlow
.ScaleTo(0.9f, 1000, Easing.OutQuint)
.FadeOut(500, Easing.OutQuint);
2023-12-28 16:14:16 +08:00
supporterDisplay
.FadeOut(500, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
}
public override void OnResuming(ScreenTransitionEvent e)
2018-04-13 17:19:50 +08:00
{
base.OnResuming(e);
2020-01-11 23:27:22 +08:00
// Ensures any playing `ButtonSystem` samples are stopped when returning to MainMenu (as to not overlap with the 'back' sample)
Buttons.StopSamplePlayback();
reappearSampleSwoosh?.Play();
ApplyToBackground(b => (b as BackgroundScreenDefault)?.Next());
2018-04-13 17:19:50 +08:00
2020-05-05 09:31:11 +08:00
// we may have consumed our preloaded instance, so let's make another.
2018-04-13 17:19:50 +08:00
preloadSongSelect();
2020-08-16 22:04:49 +08:00
musicController.EnsurePlayingSomething();
// Cycle tip on resuming
menuTip.ShowNextTip();
bottomElementsFlow
.ScaleTo(1, 1000, Easing.OutQuint)
.FadeIn(1000, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
}
public override bool OnExiting(ScreenExitEvent e)
2018-04-13 17:19:50 +08:00
{
bool requiresConfirmation =
// we need to have a dialog overlay to confirm in the first place.
dialogOverlay != null
// if the dialog has already displayed and been accepted by the user, we are good.
&& !exitConfirmedViaDialog
// Only require confirmation if there is either an ongoing operation or the user exited via a non-hold escape press.
&& (notifications.HasOngoingOperations || !exitConfirmedViaHoldOrClick);
if (requiresConfirmation)
{
2019-12-22 21:39:25 +08:00
if (dialogOverlay.CurrentDialog is ConfirmExitDialog exitDialog)
{
if (exitDialog.Buttons.OfType<PopupDialogOkButton>().FirstOrDefault() != null)
exitDialog.PerformOkAction();
else
exitDialog.Flash();
}
2019-12-22 21:39:25 +08:00
else
{
dialogOverlay.Push(new ConfirmExitDialog(() =>
{
exitConfirmedViaDialog = true;
this.Exit();
}, () =>
{
holdToExitGameOverlay.Abort();
}));
}
2021-05-19 15:27:38 +08:00
return true;
2019-12-22 20:52:00 +08:00
}
2022-04-19 14:46:03 +08:00
Buttons.State = ButtonSystemState.Exit;
OverlayActivationMode.Value = OverlayActivation.Disabled;
2020-01-11 12:17:13 +08:00
songTicker.Hide();
this.FadeOut(3000);
2023-12-28 16:19:41 +08:00
bottomElementsFlow
.FadeOut(500, Easing.OutQuint);
2023-12-28 16:14:16 +08:00
supporterDisplay
.FadeOut(500, Easing.OutQuint);
2023-12-28 16:19:41 +08:00
return base.OnExiting(e);
2018-04-13 17:19:50 +08:00
}
public void PresentBeatmap(WorkingBeatmap beatmap, RulesetInfo ruleset)
{
Logger.Log($"{nameof(MainMenu)} completing {nameof(PresentBeatmap)} with beatmap {beatmap} ruleset {ruleset}");
Beatmap.Value = beatmap;
Ruleset.Value = ruleset;
Schedule(loadSoloSongSelect);
}
public bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
{
if (e.Repeat)
return false;
switch (e.Action)
{
case GlobalAction.Back:
// In the case of a host being able to exit, the back action is handled by ExitConfirmOverlay.
Debug.Assert(!host.CanExit);
return host.SuspendToBackground();
}
return false;
}
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
{
}
2018-04-13 17:19:50 +08:00
}
}