1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-26 00:42:55 +08:00
osu-lazer/osu.Game/Overlays/Toolbar/Toolbar.cs

322 lines
13 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
2018-04-13 17:19:50 +08:00
using System;
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics;
2018-11-20 15:51:59 +08:00
using osuTK;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics.Shapes;
using osu.Framework.Allocation;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
using osu.Game.Rulesets;
using osu.Framework.Input.Bindings;
using osu.Game.Graphics.Containers;
using osu.Game.Input.Bindings;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Overlays.Toolbar
{
2022-11-24 13:32:20 +08:00
public partial class Toolbar : OverlayContainer, IKeyBindingHandler<GlobalAction>
2018-04-13 17:19:50 +08:00
{
public const float HEIGHT = 40;
public const float TOOLTIP_HEIGHT = 30;
2020-11-10 07:16:35 +08:00
/// <summary>
/// Whether the user hid this <see cref="Toolbar"/> with <see cref="GlobalAction.ToggleToolbar"/>.
2021-03-13 22:29:47 +08:00
/// In this state, automatic toggles should not occur, respecting the user's preference to have no toolbar.
2020-11-10 07:16:35 +08:00
/// </summary>
private bool hiddenByUser;
2018-04-13 17:19:50 +08:00
public Action OnHome;
2019-03-28 13:01:06 +08:00
private ToolbarUserButton userButton;
private ToolbarRulesetSelector rulesetSelector;
2018-04-13 17:19:50 +08:00
private const double transition_time = 500;
2020-08-28 02:07:24 +08:00
protected readonly IBindable<OverlayActivation> OverlayActivationMode = new Bindable<OverlayActivation>(OverlayActivation.All);
// Toolbar and its components need keyboard input even when hidden.
public override bool PropagateNonPositionalInputSubTree => OverlayActivationMode.Value != OverlayActivation.Disabled;
2019-03-08 14:14:07 +08:00
public Toolbar()
{
RelativeSizeAxes = Axes.X;
Size = new Vector2(1, HEIGHT);
AlwaysPresent = true;
}
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
// this only needed to be set for the initial LoadComplete/Update, so layout completes and gets buttons in a state they can correctly handle keyboard input for hotkeys.
AlwaysPresent = false;
2019-03-08 14:14:07 +08:00
}
[Resolved]
private Bindable<RulesetInfo> ruleset { get; set; }
2019-03-08 11:01:40 +08:00
[BackgroundDependencyLoader(true)]
private void load(OsuGame osuGame)
2018-04-13 17:19:50 +08:00
{
2022-12-30 16:07:48 +08:00
ToolbarBackground background;
HoverInterceptor interceptor;
2018-04-13 17:19:50 +08:00
Children = new Drawable[]
{
2022-12-30 16:07:48 +08:00
background = new ToolbarBackground(),
new GridContainer
2018-04-13 17:19:50 +08:00
{
RelativeSizeAxes = Axes.Both,
ColumnDimensions = new[]
2018-04-13 17:19:50 +08:00
{
new Dimension(GridSizeMode.AutoSize),
new Dimension(),
new Dimension(GridSizeMode.AutoSize)
},
Content = new[]
{
new Drawable[]
2018-04-13 17:19:50 +08:00
{
new Container
{
Name = "Left buttons",
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Depth = float.MinValue,
Children = new Drawable[]
{
new Box
{
Colour = OsuColour.Gray(0.1f),
RelativeSizeAxes = Axes.Both,
},
new FillFlowContainer
{
Direction = FillDirection.Horizontal,
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Children = new Drawable[]
{
new ToolbarSettingsButton(),
new ToolbarHomeButton
{
Action = () => OnHome?.Invoke()
},
},
},
}
},
new Container
{
Name = "Ruleset selector",
RelativeSizeAxes = Axes.Both,
Children = new Drawable[]
{
new OsuScrollContainer(Direction.Horizontal)
{
ScrollbarVisible = false,
RelativeSizeAxes = Axes.Both,
Masking = false,
Children = new Drawable[]
{
rulesetSelector = new ToolbarRulesetSelector()
}
},
new Box
{
Colour = ColourInfo.GradientHorizontal(OsuColour.Gray(0.1f).Opacity(0), OsuColour.Gray(0.1f)),
Width = 50,
RelativeSizeAxes = Axes.Y,
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
},
}
},
new Container
{
Name = "Right buttons",
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
Children = new Drawable[]
{
new Box
{
Colour = OsuColour.Gray(0.1f),
RelativeSizeAxes = Axes.Both,
},
new FillFlowContainer
{
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
Direction = FillDirection.Horizontal,
RelativeSizeAxes = Axes.Y,
AutoSizeAxes = Axes.X,
Children = new Drawable[]
{
new ToolbarNewsButton(),
new ToolbarChangelogButton(),
2023-12-28 02:11:27 +08:00
new ToolbarWikiButton(),
new ToolbarRankingsButton(),
new ToolbarBeatmapListingButton(),
new ToolbarChatButton(),
new ToolbarSocialButton(),
new ToolbarMusicButton(),
//new ToolbarButton
//{
// Icon = FontAwesome.Solid.search
//},
userButton = new ToolbarUserButton(),
new ToolbarClock(),
new ToolbarNotificationButton(),
}
},
}
},
2018-04-13 17:19:50 +08:00
},
}
2022-12-30 16:07:48 +08:00
},
interceptor = new HoverInterceptor
{
RelativeSizeAxes = Axes.Both
2018-04-13 17:19:50 +08:00
}
};
2022-12-30 16:07:48 +08:00
((IBindable<bool>)background.ShowGradient).BindTo(interceptor.ReceivedHover);
if (osuGame != null)
OverlayActivationMode.BindTo(osuGame.OverlayActivationMode);
}
protected override void LoadComplete()
{
base.LoadComplete();
rulesetSelector.Current.BindTo(ruleset);
}
2022-11-24 13:32:20 +08:00
public partial class ToolbarBackground : Container
2018-04-13 17:19:50 +08:00
{
2022-12-30 16:07:48 +08:00
public Bindable<bool> ShowGradient { get; } = new BindableBool();
2018-04-13 17:19:50 +08:00
private readonly Box gradientBackground;
public ToolbarBackground()
{
RelativeSizeAxes = Axes.Both;
Children = new Drawable[]
{
new Box
2018-04-13 17:19:50 +08:00
{
RelativeSizeAxes = Axes.Both,
Colour = OsuColour.Gray(0.1f),
},
gradientBackground = new Box
{
RelativeSizeAxes = Axes.X,
Anchor = Anchor.BottomLeft,
Alpha = 0,
Height = 80,
2018-04-13 17:19:50 +08:00
Colour = ColourInfo.GradientVertical(
OsuColour.Gray(0f).Opacity(0.7f), OsuColour.Gray(0).Opacity(0)),
2018-04-13 17:19:50 +08:00
},
};
}
protected override void LoadComplete()
{
base.LoadComplete();
2022-12-30 16:07:48 +08:00
ShowGradient.BindValueChanged(_ => updateState(), true);
2018-04-13 17:19:50 +08:00
}
private void updateState()
2018-04-13 17:19:50 +08:00
{
2022-12-30 16:07:48 +08:00
if (ShowGradient.Value)
gradientBackground.FadeIn(2500, Easing.OutQuint);
else
gradientBackground.FadeOut(200, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
}
}
2022-12-30 16:07:48 +08:00
/// <summary>
/// Whenever the mouse cursor is within the bounds of the toolbar, we want the background gradient to show, for toolbar button descriptions to be legible.
/// Unfortunately we also need to ensure that the toolbar buttons handle hover, to prevent the possibility of multiple descriptions being shown
/// due to hover events passing through multiple buttons.
/// This drawable is a workaround, that when placed front-most in the toolbar, allows to see whether hover events have been propagated through it without handling them.
/// </summary>
private partial class HoverInterceptor : Drawable
{
public IBindable<bool> ReceivedHover => receivedHover;
private readonly Bindable<bool> receivedHover = new BindableBool();
protected override bool OnHover(HoverEvent e)
{
receivedHover.Value = true;
return base.OnHover(e);
}
protected override void OnHoverLost(HoverLostEvent e)
{
receivedHover.Value = false;
base.OnHoverLost(e);
}
}
protected override void UpdateState(ValueChangedEvent<Visibility> state)
{
2021-03-13 22:29:01 +08:00
bool blockShow = hiddenByUser || OverlayActivationMode.Value == OverlayActivation.Disabled;
2021-03-13 22:29:01 +08:00
if (state.NewValue == Visibility.Visible && blockShow)
{
State.Value = Visibility.Hidden;
return;
}
base.UpdateState(state);
}
2018-04-13 17:19:50 +08:00
protected override void PopIn()
{
this.MoveToY(0, transition_time, Easing.OutQuint);
2020-07-08 14:06:40 +08:00
this.FadeIn(transition_time / 4, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
}
protected override void PopOut()
{
userButton.StateContainer?.Hide();
2018-04-13 17:19:50 +08:00
this.MoveToY(-DrawSize.Y, transition_time, Easing.OutQuint);
2020-07-08 14:06:40 +08:00
this.FadeOut(transition_time, Easing.InQuint);
2018-04-13 17:19:50 +08:00
}
2021-09-16 17:26:12 +08:00
public bool OnPressed(KeyBindingPressEvent<GlobalAction> e)
{
if (OverlayActivationMode.Value == OverlayActivation.Disabled)
return false;
2021-09-16 17:26:12 +08:00
switch (e.Action)
{
case GlobalAction.ToggleToolbar:
2021-03-13 22:29:01 +08:00
hiddenByUser = State.Value == Visibility.Visible; // set before toggling to allow the operation to always succeed.
ToggleVisibility();
return true;
}
return false;
}
2021-09-16 17:26:12 +08:00
public void OnReleased(KeyBindingReleaseEvent<GlobalAction> e)
{
}
2018-04-13 17:19:50 +08:00
}
}