1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-12 01:47:24 +08:00
osu-lazer/osu.Game/Screens/Menu/ButtonSystem.cs

332 lines
12 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;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Sprites;
using osu.Framework.Input.Bindings;
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;
using osu.Game.Input;
using osu.Game.Input.Bindings;
using osu.Game.Online.API;
2018-06-06 15:17:51 +08:00
using osu.Game.Overlays;
using osu.Game.Overlays.Notifications;
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
{
public class ButtonSystem : Container, IStateful<ButtonSystemState>, IKeyBindingHandler<GlobalAction>
2018-04-13 17:19:50 +08:00
{
public event Action<ButtonSystemState> StateChanged;
2018-04-13 17:19:50 +08:00
2018-11-26 16:07:30 +08:00
private readonly IBindable<bool> isIdle = new BindableBool();
2018-04-13 17:19:50 +08:00
public Action OnEdit;
public Action OnExit;
public Action OnDirect;
public Action OnSolo;
public Action OnSettings;
public Action OnMulti;
public Action OnChart;
public const float BUTTON_WIDTH = 140f;
public const float WEDGE_WIDTH = 20;
private OsuLogo logo;
public void SetOsuLogo(OsuLogo logo)
{
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();
}
}
private readonly Drawable iconFacade;
private readonly ButtonArea buttonArea;
2018-04-13 17:19:50 +08:00
private readonly Button backButton;
private readonly List<Button> buttonsTopLevel = new List<Button>();
private readonly List<Button> buttonsPlay = new List<Button>();
private SampleChannel sampleBack;
public ButtonSystem()
{
RelativeSizeAxes = Axes.Both;
Child = buttonArea = new ButtonArea();
buttonArea.AddRange(new[]
2018-04-13 17:19:50 +08:00
{
2019-04-02 18:55:24 +08:00
new Button(@"settings", string.Empty, FontAwesome.Solid.Cog, new Color4(85, 85, 85, 255), () => OnSettings?.Invoke(), -WEDGE_WIDTH, Key.O),
backButton = new Button(@"back", @"button-back-select", OsuIcon.LeftCircle, new Color4(51, 58, 94, 255), () => State = ButtonSystemState.TopLevel, -WEDGE_WIDTH)
{
VisibleState = ButtonSystemState.Play,
2018-04-13 17:19:50 +08:00
},
iconFacade = new Container //need a container to make the osu! icon flow properly.
{
Size = new Vector2(0, ButtonArea.BUTTON_AREA_HEIGHT)
}
});
buttonArea.Flow.CentreTarget = iconFacade;
2018-04-13 17:19:50 +08:00
}
2018-12-26 12:18:36 +08:00
[Resolved(CanBeNull = true)]
private OsuGame game { get; set; }
[Resolved]
private IAPIProvider api { get; set; }
2018-12-26 12:18:36 +08:00
[Resolved(CanBeNull = true)]
private NotificationOverlay notifications { get; set; }
2018-06-06 15:17:51 +08:00
2018-04-13 17:19:50 +08:00
[BackgroundDependencyLoader(true)]
private void load(AudioManager audio, IdleTracker idleTracker, GameHost host)
2018-04-13 17:19:50 +08:00
{
2019-04-02 18:55:24 +08:00
buttonsPlay.Add(new Button(@"solo", @"button-solo-select", FontAwesome.Solid.User, new Color4(102, 68, 204, 255), () => OnSolo?.Invoke(), WEDGE_WIDTH, Key.P));
buttonsPlay.Add(new Button(@"multi", @"button-generic-select", FontAwesome.Solid.Users, new Color4(94, 63, 186, 255), onMulti, 0, Key.M));
buttonsPlay.Add(new Button(@"chart", @"button-generic-select", OsuIcon.Charts, new Color4(80, 53, 160, 255), () => OnChart?.Invoke()));
buttonsPlay.ForEach(b => b.VisibleState = ButtonSystemState.Play);
buttonsTopLevel.Add(new Button(@"play", @"button-play-select", OsuIcon.Logo, new Color4(102, 68, 204, 255), () => State = ButtonSystemState.Play, WEDGE_WIDTH, Key.P));
buttonsTopLevel.Add(new Button(@"osu!editor", @"button-generic-select", OsuIcon.EditCircle, new Color4(238, 170, 0, 255), () => OnEdit?.Invoke(), 0, Key.E));
buttonsTopLevel.Add(new Button(@"osu!direct", @"button-direct-select", OsuIcon.ChevronDownCircle, new Color4(165, 204, 0, 255), () => OnDirect?.Invoke(), 0, Key.D));
if (host.CanExit)
buttonsTopLevel.Add(new Button(@"exit", string.Empty, OsuIcon.CrossCircle, new Color4(238, 51, 153, 255), () => OnExit?.Invoke(), 0, Key.Q));
buttonArea.AddRange(buttonsPlay);
buttonArea.AddRange(buttonsTopLevel);
isIdle.ValueChanged += idle => updateIdleState(idle.NewValue);
2018-11-26 15:50:41 +08:00
if (idleTracker != null) isIdle.BindTo(idleTracker.IsIdle);
2018-08-31 06:04:40 +08:00
sampleBack = audio.Sample.Get(@"Menu/button-back-select");
2018-04-13 17:19:50 +08:00
}
private void onMulti()
{
if (!api.IsLoggedIn)
{
2018-12-26 12:18:36 +08:00
notifications?.Post(new SimpleNotification
{
Text = "You gotta be logged in to multi 'yo!",
2019-04-02 18:55:24 +08:00
Icon = FontAwesome.Solid.Globe
});
return;
}
2018-12-26 12:18:36 +08:00
OnMulti?.Invoke();
}
private void updateIdleState(bool isIdle)
{
if (isIdle && State != ButtonSystemState.Exit)
State = ButtonSystemState.Initial;
}
public bool OnPressed(GlobalAction action)
2018-04-30 01:15:09 +08:00
{
switch (action)
{
case GlobalAction.Back:
return goBack();
case GlobalAction.Select:
logo?.Click();
return true;
default:
return false;
}
}
public bool OnReleased(GlobalAction action) => false;
private bool goBack()
{
switch (State)
{
case ButtonSystemState.TopLevel:
State = ButtonSystemState.Initial;
sampleBack?.Play();
return true;
case ButtonSystemState.Play:
backButton.Click();
return true;
default:
return false;
}
2018-04-30 01:15:09 +08:00
}
2018-04-13 17:19:50 +08:00
private bool onOsuLogo()
{
switch (state)
{
default:
return true;
case ButtonSystemState.Initial:
State = ButtonSystemState.TopLevel;
2018-04-13 17:19:50 +08:00
return true;
case ButtonSystemState.TopLevel:
buttonsTopLevel.First().Click();
2018-04-13 17:19:50 +08:00
return false;
case ButtonSystemState.Play:
buttonsPlay.First().Click();
2018-04-13 17:19:50 +08:00
return false;
}
}
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;
if (game != null)
game.OverlayActivationMode.Value = state == ButtonSystemState.Exit ? OverlayActivation.Disabled : OverlayActivation.All;
2018-04-13 17:19:50 +08:00
updateLogoState(lastState);
Logger.Log($"{nameof(ButtonSystem)}'s state changed from {lastState} to {state}");
2018-04-13 17:19:50 +08:00
using (buttonArea.BeginDelayedSequence(lastState == ButtonSystemState.Initial ? 150 : 0, true))
{
buttonArea.ButtonSystemState = state;
2018-04-13 17:19:50 +08:00
foreach (var b in buttonArea.Children.OfType<Button>())
b.ButtonSystemState = state;
2018-04-13 17:19:50 +08:00
}
StateChanged?.Invoke(State);
}
}
private ScheduledDelegate logoDelayedAction;
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(() =>
2018-06-06 17:21:03 +08:00
{
logoTracking = false;
game?.Toolbar.Hide();
2018-06-06 15:17:51 +08:00
2018-06-06 17:21:03 +08:00
logo.ClearTransforms(targetMember: nameof(Position));
logo.RelativePositionAxes = Axes.Both;
2018-04-13 17:19:50 +08:00
2018-06-06 17:21:03 +08:00
logo.MoveTo(new Vector2(0.5f), 800, Easing.OutExpo);
logo.ScaleTo(1, 800, Easing.OutExpo);
}, buttonArea.Alpha * 150);
2018-04-13 17:19:50 +08:00
break;
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;
case ButtonSystemState.Initial:
logo.ClearTransforms(targetMember: nameof(Position));
logo.RelativePositionAxes = Axes.None;
bool impact = logo.Scale.X > 0.6f;
if (lastState == ButtonSystemState.Initial)
logo.ScaleTo(0.5f, 200, Easing.In);
2018-04-13 17:19:50 +08:00
logo.MoveTo(logoTrackingPosition, lastState == ButtonSystemState.EnteringMode ? 0 : 200, Easing.In);
2018-04-13 17:19:50 +08:00
logoDelayedAction?.Cancel();
2018-04-13 17:19:50 +08:00
logoDelayedAction = Scheduler.AddDelayed(() =>
{
logoTracking = true;
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;
default:
logo.ClearTransforms(targetMember: nameof(Position));
logo.RelativePositionAxes = Axes.None;
2018-04-13 17:19:50 +08:00
logoTracking = true;
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;
case ButtonSystemState.EnteringMode:
2018-04-13 17:19:50 +08:00
logoTracking = true;
break;
}
}
private Vector2 logoTrackingPosition => logo.Parent.ToLocalSpace(iconFacade.ScreenSpaceDrawQuad.Centre);
private bool logoTracking;
protected override void Update()
{
base.Update();
if (logo != null)
{
if (logoTracking && logo.RelativePositionAxes == Axes.None && iconFacade.IsLoaded)
2018-04-13 17:19:50 +08:00
logo.Position = logoTrackingPosition;
iconFacade.Width = logo.SizeForFlow * 0.5f;
}
}
}
public enum ButtonSystemState
2018-04-13 17:19:50 +08:00
{
Exit,
2018-04-13 17:19:50 +08:00
Initial,
TopLevel,
Play,
EnteringMode,
}
}