1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-23 03:02:55 +08:00
osu-lazer/osu.Game/Screens/Menu/MainMenu.cs

310 lines
9.5 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;
2019-12-22 20:52:00 +08:00
using System.Linq;
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation;
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.Sprites;
using osu.Framework.Platform;
2018-04-13 17:19:50 +08:00
using osu.Framework.Screens;
using osu.Game.Configuration;
using osu.Game.Graphics;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics.Containers;
2019-09-08 13:36:58 +08:00
using osu.Game.Online.API;
using osu.Game.Overlays;
using osu.Game.Overlays.Dialog;
2018-04-13 17:19:50 +08:00
using osu.Game.Screens.Backgrounds;
using osu.Game.Screens.Edit;
2018-05-16 08:14:10 +08:00
using osu.Game.Screens.Multi;
2018-04-13 17:19:50 +08:00
using osu.Game.Screens.Select;
namespace osu.Game.Screens.Menu
{
public class MainMenu : OsuScreen
{
public const float FADE_IN_DURATION = 300;
public const float FADE_OUT_DURATION = 400;
2018-04-13 17:19:50 +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 AllowRateAdjustments => false;
2018-04-13 17:19:50 +08:00
private Screen songSelect;
private MenuSideFlashes sideFlashes;
2018-04-13 17:19:50 +08:00
private ButtonSystem buttons;
[Resolved]
private GameHost host { get; set; }
2018-04-13 17:19:50 +08:00
[Resolved(canBeNull: true)]
private MusicController music { get; set; }
2019-09-08 13:36:58 +08:00
[Resolved(canBeNull: true)]
private LoginOverlay login { get; set; }
[Resolved]
private IAPIProvider api { get; set; }
2019-09-19 16:40:46 +08:00
[Resolved(canBeNull: true)]
private DialogOverlay dialogOverlay { get; set; }
2019-03-11 23:04:19 +08:00
private BackgroundScreenDefault background;
protected override BackgroundScreen CreateBackground() => background;
2018-04-13 17:19:50 +08:00
2019-10-02 12:26:46 +08:00
private Bindable<float> holdDelay;
2019-09-25 20:56:47 +08:00
private Bindable<bool> loginDisplayed;
2019-09-19 19:17:58 +08:00
private ExitConfirmOverlay exitConfirmOverlay;
private ParallaxContainer buttonsContainer;
2020-01-11 12:17:13 +08:00
private SongTicker songTicker;
[BackgroundDependencyLoader(true)]
private void load(BeatmapListingOverlay beatmapListing, SettingsOverlay settings, RankingsOverlay rankings, OsuConfigManager config, SessionStatics statics)
2018-04-13 17:19:50 +08:00
{
2019-10-02 12:26:46 +08:00
holdDelay = config.GetBindable<float>(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(exitConfirmOverlay = new ExitConfirmOverlay
{
Action = () =>
{
if (holdDelay.Value > 0)
confirmAndExit();
else
this.Exit();
}
});
}
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[]
{
buttons = new ButtonSystem
{
OnEdit = delegate { this.Push(new Editor()); },
OnSolo = onSolo,
OnMulti = delegate { this.Push(new Multiplayer()); },
2019-09-19 21:46:21 +08:00
OnExit = confirmAndExit,
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
},
exitConfirmOverlay?.CreateProxy() ?? Drawable.Empty()
});
buttons.StateChanged += state =>
{
switch (state)
{
case ButtonSystemState.Initial:
case ButtonSystemState.Exit:
2019-01-23 19:52:00 +08:00
Background.FadeColour(Color4.White, 500, Easing.OutSine);
break;
2019-04-01 11:44:46 +08:00
default:
2019-01-23 19:52:00 +08:00
Background.FadeColour(OsuColour.Gray(0.8f), 500, Easing.OutSine);
break;
}
};
2018-04-13 17:19:50 +08:00
buttons.OnSettings = () => settings?.ToggleVisibility();
buttons.OnBeatmapListing = () => beatmapListing?.ToggleVisibility();
buttons.OnChart = () => rankings?.ShowSpotlights();
2018-04-13 17:19:50 +08:00
2019-03-11 23:04:19 +08:00
LoadComponentAsync(background = new BackgroundScreenDefault());
2018-04-13 17:19:50 +08:00
preloadSongSelect();
}
[Resolved(canBeNull: true)]
private OsuGame game { get; set; }
2019-09-19 21:46:21 +08:00
private void confirmAndExit()
{
2019-12-22 20:52:00 +08:00
if (exitConfirmed) return;
2019-09-19 21:46:21 +08:00
exitConfirmed = true;
game?.PerformFromScreen(menu => menu.Exit());
2019-09-19 21:46:21 +08:00
}
2018-04-13 17:19:50 +08:00
private void preloadSongSelect()
{
if (songSelect == null)
LoadComponentAsync(songSelect = new PlaySongSelect());
}
public void LoadToSolo() => Schedule(onSolo);
2019-02-25 18:19:28 +08:00
private void onSolo() => this.Push(consumeSongSelect());
2018-04-13 17:19:50 +08:00
private Screen consumeSongSelect()
{
var s = songSelect;
songSelect = null;
return s;
}
2019-01-23 19:52:00 +08:00
public override void OnEntering(IScreen last)
2018-04-13 17:19:50 +08:00
{
base.OnEntering(last);
buttons.FadeInFromZero(500);
var track = Beatmap.Value.Track;
var metadata = Beatmap.Value.Metadata;
if (last is IntroScreen && track != null)
2018-04-13 17:19:50 +08:00
{
if (!track.IsRunning)
{
track.Seek(metadata.PreviewTime != -1 ? metadata.PreviewTime : 0.4f * track.Length);
track.Start();
}
}
}
private bool exitConfirmed;
2018-04-13 17:19:50 +08:00
protected override void LogoArriving(OsuLogo logo, bool resuming)
{
base.LogoArriving(logo, resuming);
buttons.SetOsuLogo(logo);
logo.FadeColour(Color4.White, 100, Easing.OutQuint);
logo.FadeIn(100, Easing.OutQuint);
if (resuming)
{
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)
{
logo.Action += displayLogin;
}
bool displayLogin()
{
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 true;
}
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);
seq.OnComplete(_ => buttons.SetOsuLogo(null));
seq.OnAbort(_ => buttons.SetOsuLogo(null));
2018-04-13 17:19:50 +08:00
}
2019-01-23 19:52:00 +08:00
public override void OnSuspending(IScreen next)
2018-04-13 17:19:50 +08:00
{
base.OnSuspending(next);
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);
}
2019-01-23 19:52:00 +08:00
public override void OnResuming(IScreen last)
2018-04-13 17:19:50 +08:00
{
base.OnResuming(last);
2020-01-11 23:27:22 +08:00
2019-02-25 18:19:28 +08:00
(Background 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();
if (Beatmap.Value.Track != null && music?.IsUserPaused != true)
Beatmap.Value.Track.Start();
2018-04-13 17:19:50 +08:00
}
2019-01-23 19:52:00 +08:00
public override bool OnExiting(IScreen next)
2018-04-13 17:19:50 +08:00
{
if (!exitConfirmed && dialogOverlay != null)
{
2019-12-22 21:39:25 +08:00
if (dialogOverlay.CurrentDialog is ConfirmExitDialog exitDialog)
{
exitConfirmed = true;
exitDialog.Buttons.First().Click();
}
else
{
dialogOverlay.Push(new ConfirmExitDialog(confirmAndExit, () => exitConfirmOverlay.Abort()));
return true;
}
2019-12-22 20:52:00 +08:00
}
buttons.State = ButtonSystemState.Exit;
2020-01-11 12:17:13 +08:00
songTicker.Hide();
this.FadeOut(3000);
2018-04-13 17:19:50 +08:00
return base.OnExiting(next);
}
2019-09-19 19:17:58 +08:00
private class ConfirmExitDialog : PopupDialog
{
2019-09-19 19:17:58 +08:00
public ConfirmExitDialog(Action confirm, Action cancel)
{
HeaderText = "Are you sure you want to exit?";
BodyText = "Last chance to back out.";
Icon = FontAwesome.Solid.ExclamationTriangle;
Buttons = new PopupDialogButton[]
{
new PopupDialogOkButton
{
2020-04-08 08:34:18 +08:00
Text = @"Goodbye",
Action = confirm
},
new PopupDialogCancelButton
{
2019-09-19 19:17:58 +08:00
Text = @"Just a little more",
Action = cancel
},
};
}
}
2018-04-13 17:19:50 +08:00
}
}