1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-29 02:52:54 +08:00
osu-lazer/osu.Game/Overlays/Mods/ModButton.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

320 lines
10 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
2018-11-20 15:51:59 +08:00
using osuTK;
using osuTK.Graphics;
using osuTK.Input;
2017-02-23 10:16:23 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Sprites;
using osu.Game.Graphics.Sprites;
2017-04-18 15:05:58 +08:00
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.UI;
using System;
using System.Linq;
using osu.Framework.Graphics.Cursor;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
using osu.Framework.Localisation;
using osu.Game.Graphics;
2017-11-26 02:28:11 +08:00
using osu.Game.Graphics.UserInterface;
2018-04-13 17:19:50 +08:00
2017-02-23 10:16:23 +08:00
namespace osu.Game.Overlays.Mods
{
/// <summary>
/// Represents a clickable button which can cycle through one of more mods.
/// </summary>
public class ModButton : ModButtonEmpty, IHasCustomTooltip<Mod>
2017-02-23 10:16:23 +08:00
{
2017-05-08 18:56:04 +08:00
private ModIcon foregroundIcon;
2017-05-26 22:10:28 +08:00
private ModIcon backgroundIcon;
private readonly SpriteText text;
private readonly Container<ModIcon> iconsContainer;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Fired when the selection changes.
/// </summary>
public Action<Mod> SelectionChanged;
2018-04-13 17:19:50 +08:00
public LocalisableString TooltipText => (SelectedMod?.Description ?? Mods.FirstOrDefault()?.Description) ?? string.Empty;
2018-04-13 17:19:50 +08:00
2017-07-23 02:50:25 +08:00
private const Easing mod_switch_easing = Easing.InOutSine;
private const double mod_switch_duration = 120;
2018-04-13 17:19:50 +08:00
// A selected index of -1 means not selected.
private int selectedIndex = -1;
2018-04-13 17:19:50 +08:00
/// <summary>
/// Change the selected mod index of this button.
/// </summary>
2018-01-03 12:12:49 +08:00
/// <param name="newIndex">The new index.</param>
/// <param name="resetSettings">Whether any settings applied to the mod should be reset on selection.</param>
/// <returns>Whether the selection changed.</returns>
private bool changeSelectedIndex(int newIndex, bool resetSettings = true)
2017-02-23 10:16:23 +08:00
{
2018-01-02 15:55:03 +08:00
if (newIndex == selectedIndex) return false;
2018-04-13 17:19:50 +08:00
2018-01-02 15:55:03 +08:00
int direction = newIndex < selectedIndex ? -1 : 1;
bool beforeSelected = Selected;
2018-04-13 17:19:50 +08:00
Mod previousSelection = SelectedMod ?? Mods[0];
2018-04-13 17:19:50 +08:00
2018-01-02 15:55:03 +08:00
if (newIndex >= Mods.Length)
newIndex = -1;
else if (newIndex < -1)
newIndex = Mods.Length - 1;
2018-04-13 17:19:50 +08:00
2018-01-02 15:55:03 +08:00
if (newIndex >= 0 && !Mods[newIndex].HasImplementation)
return false;
2018-04-13 17:19:50 +08:00
selectedIndex = newIndex;
Mod newSelection = SelectedMod ?? Mods[0];
if (resetSettings)
newSelection.ResetSettingsToDefaults();
Schedule(() =>
{
if (beforeSelected != Selected)
{
iconsContainer.RotateTo(Selected ? 5f : 0f, 300, Easing.OutElastic);
iconsContainer.ScaleTo(Selected ? 1.1f : 1f, 300, Easing.OutElastic);
}
2018-04-13 17:19:50 +08:00
if (previousSelection != newSelection)
{
const float rotate_angle = 16;
2018-04-13 17:19:50 +08:00
foregroundIcon.RotateTo(rotate_angle * direction, mod_switch_duration, mod_switch_easing);
backgroundIcon.RotateTo(-rotate_angle * direction, mod_switch_duration, mod_switch_easing);
2019-04-01 11:16:05 +08:00
backgroundIcon.Mod = newSelection;
2021-07-05 23:52:39 +08:00
using (BeginDelayedSequence(mod_switch_duration))
{
foregroundIcon
.RotateTo(-rotate_angle * direction)
.RotateTo(0f, mod_switch_duration, mod_switch_easing);
2018-04-13 17:19:50 +08:00
backgroundIcon
.RotateTo(rotate_angle * direction)
.RotateTo(0f, mod_switch_duration, mod_switch_easing);
2018-04-13 17:19:50 +08:00
Schedule(() => DisplayMod(newSelection));
}
2017-05-26 22:10:28 +08:00
}
2018-04-13 17:19:50 +08:00
foregroundIcon.Selected.Value = Selected;
});
2018-04-13 17:19:50 +08:00
SelectionChanged?.Invoke(SelectedMod);
return true;
2017-02-23 10:16:23 +08:00
}
2018-04-13 17:19:50 +08:00
public bool Selected => selectedIndex != -1;
2018-04-13 17:19:50 +08:00
2017-02-23 10:16:23 +08:00
private Color4 selectedColour;
2018-04-13 17:19:50 +08:00
2017-02-23 10:16:23 +08:00
public Color4 SelectedColour
{
get => selectedColour;
2017-02-23 10:16:23 +08:00
set
{
if (value == selectedColour) return;
2019-02-28 12:31:40 +08:00
2017-02-23 10:16:23 +08:00
selectedColour = value;
2017-03-06 19:03:26 +08:00
if (Selected) foregroundIcon.Colour = value;
2017-02-23 10:16:23 +08:00
}
}
2018-04-13 17:19:50 +08:00
private Mod mod;
protected readonly Container ButtonContent;
2018-04-13 17:19:50 +08:00
public Mod Mod
2017-02-23 10:16:23 +08:00
{
get => mod;
2017-02-23 10:16:23 +08:00
set
{
mod = value;
2018-04-13 17:19:50 +08:00
if (mod == null)
{
2017-09-24 03:45:46 +08:00
Mods = Array.Empty<Mod>();
Alpha = 0;
}
else
{
Mods = (mod as MultiMod)?.Mods ?? new[] { mod };
Alpha = 1;
}
2018-04-13 17:19:50 +08:00
2017-02-23 10:16:23 +08:00
createIcons();
2019-04-01 11:16:05 +08:00
2017-03-07 09:59:19 +08:00
if (Mods.Length > 0)
2017-02-23 10:16:23 +08:00
{
DisplayMod(Mods[0]);
2017-02-23 10:16:23 +08:00
}
}
}
2018-04-13 17:19:50 +08:00
2017-03-07 09:59:19 +08:00
public Mod[] Mods { get; private set; }
2018-04-13 17:19:50 +08:00
public virtual Mod SelectedMod => Mods.ElementAtOrDefault(selectedIndex);
2018-04-13 17:19:50 +08:00
2018-10-02 11:02:47 +08:00
protected override bool OnMouseDown(MouseDownEvent e)
{
ButtonContent.ScaleTo(0.9f, 800, Easing.Out);
2018-10-02 11:02:47 +08:00
return base.OnMouseDown(e);
}
protected override void OnMouseUp(MouseUpEvent e)
2017-02-23 10:16:23 +08:00
{
ButtonContent.ScaleTo(1, 500, Easing.OutElastic);
// only trigger the event if we are inside the area of the button
if (Contains(e.ScreenSpaceMousePosition))
{
2018-10-02 11:02:47 +08:00
switch (e.Button)
{
case MouseButton.Right:
SelectNext(-1);
break;
}
}
2017-02-23 10:16:23 +08:00
}
2018-04-13 17:19:50 +08:00
protected override bool OnClick(ClickEvent e)
{
SelectNext(1);
2019-12-09 01:12:32 +08:00
return true;
}
2018-01-02 15:55:03 +08:00
/// <summary>
/// Select the next available mod in a specified direction.
/// </summary>
/// <param name="direction">1 for forwards, -1 for backwards.</param>
public void SelectNext(int direction)
{
int start = selectedIndex + direction;
// wrap around if we are at an extremity.
if (start >= Mods.Length)
start = -1;
else if (start < -1)
start = Mods.Length - 1;
2018-04-13 17:19:50 +08:00
2018-01-02 15:55:03 +08:00
for (int i = start; i < Mods.Length && i >= 0; i += direction)
2019-11-11 19:53:22 +08:00
{
if (SelectAt(i))
return;
2019-11-11 19:53:22 +08:00
}
2018-04-13 17:19:50 +08:00
2018-01-02 15:55:03 +08:00
Deselect();
}
2018-04-13 17:19:50 +08:00
/// <summary>
/// Select the mod at the provided index.
/// </summary>
/// <param name="index">The index to select.</param>
/// <param name="resetSettings">Whether any settings applied to the mod should be reset on selection.</param>
/// <returns>Whether the selection changed.</returns>
public bool SelectAt(int index, bool resetSettings = true)
{
if (!Mods[index].HasImplementation) return false;
2018-04-13 17:19:50 +08:00
changeSelectedIndex(index, resetSettings);
return true;
}
2018-04-13 17:19:50 +08:00
public void Deselect() => changeSelectedIndex(-1);
2018-04-13 17:19:50 +08:00
protected virtual void DisplayMod(Mod mod)
2017-02-23 10:16:23 +08:00
{
2017-05-29 17:10:02 +08:00
if (backgroundIcon != null)
2020-01-14 20:11:32 +08:00
backgroundIcon.Mod = foregroundIcon.Mod;
foregroundIcon.Mod = mod;
text.Text = mod.Name;
Colour = mod.HasImplementation ? Color4.White : Color4.Gray;
2017-02-23 10:16:23 +08:00
}
2018-04-13 17:19:50 +08:00
2017-02-23 10:16:23 +08:00
private void createIcons()
{
2017-03-06 19:03:26 +08:00
iconsContainer.Clear();
2019-04-01 11:16:05 +08:00
2017-02-23 10:16:23 +08:00
if (Mods.Length > 1)
{
2017-07-11 21:58:06 +08:00
iconsContainer.AddRange(new[]
2017-02-23 10:16:23 +08:00
{
2021-01-26 18:11:19 +08:00
backgroundIcon = new ModIcon(Mods[1], false)
2017-02-23 10:16:23 +08:00
{
2017-05-26 22:10:28 +08:00
Origin = Anchor.BottomRight,
Anchor = Anchor.BottomRight,
2017-02-23 10:16:23 +08:00
Position = new Vector2(1.5f),
},
2021-01-26 18:11:19 +08:00
foregroundIcon = new ModIcon(Mods[0], false)
2017-02-23 10:16:23 +08:00
{
2017-05-26 22:10:28 +08:00
Origin = Anchor.BottomRight,
Anchor = Anchor.BottomRight,
2017-02-23 10:16:23 +08:00
Position = new Vector2(-1.5f),
},
});
}
else
{
2021-01-26 18:11:19 +08:00
iconsContainer.Add(foregroundIcon = new ModIcon(Mod, false)
2017-02-23 10:16:23 +08:00
{
2017-03-06 19:03:26 +08:00
Origin = Anchor.Centre,
Anchor = Anchor.Centre,
2017-02-23 10:16:23 +08:00
});
}
}
2018-04-13 17:19:50 +08:00
public ModButton(Mod mod)
2017-03-07 00:46:36 +08:00
{
2017-02-23 10:16:23 +08:00
Children = new Drawable[]
{
new Container
{
Size = new Vector2(77f, 80f),
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Children = new Drawable[]
{
ButtonContent = new Container
2017-02-23 10:16:23 +08:00
{
Children = new Drawable[]
{
iconsContainer = new Container<ModIcon>
{
RelativeSizeAxes = Axes.Both,
Origin = Anchor.Centre,
Anchor = Anchor.Centre,
},
},
2017-02-23 10:16:23 +08:00
RelativeSizeAxes = Axes.Both,
Origin = Anchor.Centre,
Anchor = Anchor.Centre,
}
}
},
text = new OsuSpriteText
{
Y = 75,
2017-02-23 10:16:23 +08:00
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Font = OsuFont.GetFont(size: 18)
2017-02-23 10:16:23 +08:00
},
2021-06-18 20:14:51 +08:00
new HoverSounds()
2017-02-23 10:16:23 +08:00
};
Mod = mod;
2017-02-23 10:16:23 +08:00
}
public virtual ITooltip<Mod> GetCustomTooltip() => new ModButtonTooltip();
public Mod TooltipContent => SelectedMod ?? Mods.FirstOrDefault();
2017-02-23 10:16:23 +08:00
}
}