1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-26 02:52:54 +08:00
osu-lazer/osu.Game/Screens/Edit/Components/PlaybackControl.cs

233 lines
8.1 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.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;
using osu.Framework.Audio;
using osu.Framework.Bindables;
2018-04-13 17:19:50 +08:00
using osu.Framework.Extensions.IEnumerableExtensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Cursor;
using osu.Framework.Graphics.Sprites;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics.UserInterface;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
using osu.Framework.Localisation;
2018-04-13 17:19:50 +08:00
using osu.Game.Graphics;
using osu.Game.Graphics.Sprites;
using osu.Game.Graphics.UserInterface;
2023-01-15 06:50:41 +08:00
using osu.Game.Localisation;
using osu.Game.Overlays;
2018-11-30 13:41:17 +08:00
using osuTK.Input;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Screens.Edit.Components
{
2022-11-24 13:32:20 +08:00
public partial class PlaybackControl : BottomBarContainer
2018-04-13 17:19:50 +08:00
{
private IconButton playButton = null!;
private PlaybackSpeedControl playbackSpeedControl = null!;
2018-04-13 17:19:50 +08:00
2020-02-14 21:14:00 +08:00
[Resolved]
private EditorClock editorClock { get; set; } = null!;
2018-04-13 17:19:50 +08:00
private readonly Bindable<EditorScreenMode> currentScreenMode = new Bindable<EditorScreenMode>();
private readonly BindableNumber<double> tempoAdjustment = new BindableDouble(1);
2018-04-13 17:19:50 +08:00
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colourProvider, Editor? editor)
2018-04-13 17:19:50 +08:00
{
Background.Colour = colourProvider.Background4;
2018-04-13 17:19:50 +08:00
Children = new Drawable[]
{
playButton = new IconButton
{
Anchor = Anchor.CentreLeft,
2019-06-16 02:05:58 +08:00
Origin = Anchor.CentreLeft,
Scale = new Vector2(1.2f),
IconScale = new Vector2(1.2f),
2019-04-02 18:55:24 +08:00
Icon = FontAwesome.Regular.PlayCircle,
2018-04-13 17:19:50 +08:00
Action = togglePause,
},
playbackSpeedControl = new PlaybackSpeedControl
2018-04-13 17:19:50 +08:00
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Padding = new MarginPadding { Left = 45, },
Anchor = Anchor.CentreRight,
Origin = Anchor.CentreRight,
Direction = FillDirection.Vertical,
Children = new Drawable[]
{
new OsuSpriteText
{
Text = EditorStrings.PlaybackSpeed,
},
new PlaybackTabControl
{
Current = tempoAdjustment,
RelativeSizeAxes = Axes.X,
Height = 16,
},
}
2018-04-13 17:19:50 +08:00
}
};
Track.BindValueChanged(tr => tr.NewValue?.AddAdjustment(AdjustableProperty.Tempo, tempoAdjustment), true);
if (editor != null)
currentScreenMode.BindTo(editor.Mode);
}
protected override void LoadComplete()
{
base.LoadComplete();
currentScreenMode.BindValueChanged(_ =>
{
if (currentScreenMode.Value == EditorScreenMode.Timing)
{
tempoAdjustment.Value = 1;
tempoAdjustment.Disabled = true;
playbackSpeedControl.FadeTo(0.5f, 400, Easing.OutQuint);
playbackSpeedControl.TooltipText = "Speed adjustment is unavailable in timing mode. Timing at slower speeds is inaccurate due to resampling artifacts.";
}
else
{
tempoAdjustment.Disabled = false;
playbackSpeedControl.FadeTo(1, 400, Easing.OutQuint);
playbackSpeedControl.TooltipText = default;
}
});
}
protected override void Dispose(bool isDisposing)
{
Track.Value?.RemoveAdjustment(AdjustableProperty.Tempo, tempoAdjustment);
base.Dispose(isDisposing);
2018-04-13 17:19:50 +08:00
}
2018-11-30 13:41:17 +08:00
protected override bool OnKeyDown(KeyDownEvent e)
{
if (e.Repeat)
return false;
2018-11-30 13:41:17 +08:00
switch (e.Key)
{
case Key.Space:
togglePause();
return true;
}
return base.OnKeyDown(e);
}
2018-04-13 17:19:50 +08:00
private void togglePause()
{
if (editorClock.IsRunning)
editorClock.Stop();
2018-04-13 17:19:50 +08:00
else
editorClock.Start();
2018-04-13 17:19:50 +08:00
}
2024-02-21 00:31:28 +08:00
private static readonly IconUsage play_icon = FontAwesome.Regular.PlayCircle;
private static readonly IconUsage pause_icon = FontAwesome.Regular.PauseCircle;
2018-04-13 17:19:50 +08:00
protected override void Update()
{
base.Update();
2024-02-21 00:31:28 +08:00
playButton.Icon = editorClock.IsRunning ? pause_icon : play_icon;
2018-04-13 17:19:50 +08:00
}
private partial class PlaybackSpeedControl : FillFlowContainer, IHasTooltip
{
public LocalisableString TooltipText { get; set; }
}
2022-11-24 13:32:20 +08:00
private partial class PlaybackTabControl : OsuTabControl<double>
2018-04-13 17:19:50 +08:00
{
private static readonly double[] tempo_values = { 0.25, 0.5, 0.75, 1 };
2018-04-13 17:19:50 +08:00
protected override TabItem<double> CreateTabItem(double value) => new PlaybackTabItem(value);
protected override Dropdown<double> CreateDropdown() => null!;
2018-04-13 17:19:50 +08:00
public PlaybackTabControl()
{
RelativeSizeAxes = Axes.Both;
TabContainer.Spacing = Vector2.Zero;
tempo_values.ForEach(AddItem);
Current.Value = tempo_values.Last();
2018-04-13 17:19:50 +08:00
}
2022-11-24 13:32:20 +08:00
public partial class PlaybackTabItem : TabItem<double>
2018-04-13 17:19:50 +08:00
{
private const float fade_duration = 200;
private readonly OsuSpriteText text;
private readonly OsuSpriteText textBold;
2019-02-28 12:31:40 +08:00
public PlaybackTabItem(double value)
: base(value)
2018-04-13 17:19:50 +08:00
{
RelativeSizeAxes = Axes.Both;
Width = 1f / tempo_values.Length;
Children = new Drawable[]
{
text = new OsuSpriteText
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Text = $"{value:0%}",
Font = OsuFont.GetFont(size: 14)
2018-04-13 17:19:50 +08:00
},
textBold = new OsuSpriteText
{
Origin = Anchor.TopCentre,
Anchor = Anchor.TopCentre,
Text = $"{value:0%}",
Font = OsuFont.GetFont(size: 14, weight: FontWeight.Bold),
2018-04-13 17:19:50 +08:00
Alpha = 0,
},
};
}
private Color4 hoveredColour;
private Color4 normalColour;
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colourProvider)
2018-04-13 17:19:50 +08:00
{
text.Colour = normalColour = colourProvider.Light3;
textBold.Colour = hoveredColour = colourProvider.Content1;
2018-04-13 17:19:50 +08:00
}
2018-10-02 11:02:47 +08:00
protected override bool OnHover(HoverEvent e)
2018-04-13 17:19:50 +08:00
{
updateState();
return false;
2018-04-13 17:19:50 +08:00
}
2018-10-02 11:02:47 +08:00
protected override void OnHoverLost(HoverLostEvent e) => updateState();
2018-04-13 17:19:50 +08:00
protected override void OnActivated() => updateState();
protected override void OnDeactivated() => updateState();
private void updateState()
{
2019-02-21 17:56:34 +08:00
text.FadeColour(Active.Value || IsHovered ? hoveredColour : normalColour, fade_duration, Easing.OutQuint);
text.FadeTo(Active.Value ? 0 : 1, fade_duration, Easing.OutQuint);
textBold.FadeTo(Active.Value ? 1 : 0, fade_duration, Easing.OutQuint);
2018-04-13 17:19:50 +08:00
}
}
}
}
}