1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-16 12:37:47 +08:00
osu-lazer/osu.Game/Overlays/VolumeOverlay.cs

229 lines
7.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
2022-06-17 15:37:17 +08:00
#nullable disable
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
2019-02-21 18:04:31 +08:00
using osu.Framework.Bindables;
2018-04-13 17:19:50 +08:00
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
2018-10-02 11:02:47 +08:00
using osu.Framework.Input.Events;
2018-04-13 17:19:50 +08:00
using osu.Framework.Threading;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
2018-04-13 17:19:50 +08:00
using osu.Game.Input.Bindings;
using osu.Game.Overlays.Volume;
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.Overlays
{
2022-11-24 13:32:20 +08:00
public partial class VolumeOverlay : VisibilityContainer
2018-04-13 17:19:50 +08:00
{
private const float offset = 10;
private VolumeMeter volumeMeterMaster;
private VolumeMeter volumeMeterEffect;
private VolumeMeter volumeMeterMusic;
private MuteButton muteButton;
private readonly BindableDouble muteAdjustment = new BindableDouble();
2019-11-12 17:45:42 +08:00
public Bindable<bool> IsMuted { get; } = new Bindable<bool>();
2019-09-15 22:31:57 +08:00
private SelectionCycleFillFlowContainer<VolumeMeter> volumeMeters;
2018-04-13 17:19:50 +08:00
[BackgroundDependencyLoader]
private void load(AudioManager audio, OsuColour colours)
{
AutoSizeAxes = Axes.X;
RelativeSizeAxes = Axes.Y;
AddRange(new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Y,
Width = 300,
Colour = ColourInfo.GradientHorizontal(Color4.Black.Opacity(0.75f), Color4.Black.Opacity(0))
},
muteButton = new MuteButton
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Margin = new MarginPadding(10),
Current = { BindTarget = IsMuted }
},
volumeMeters = new SelectionCycleFillFlowContainer<VolumeMeter>
2018-04-13 17:19:50 +08:00
{
Direction = FillDirection.Vertical,
AutoSizeAxes = Axes.Both,
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Spacing = new Vector2(0, offset),
Margin = new MarginPadding { Left = offset },
2021-07-04 22:06:28 +08:00
Children = new[]
2018-04-13 17:19:50 +08:00
{
volumeMeterEffect = new VolumeMeter("EFFECTS", 125, colours.BlueDarker),
volumeMeterMaster = new VolumeMeter("MASTER", 150, colours.PinkDarker),
volumeMeterMusic = new VolumeMeter("MUSIC", 125, colours.BlueDarker),
2018-04-13 17:19:50 +08:00
}
}
2018-04-13 17:19:50 +08:00
});
volumeMeterMaster.Bindable.BindTo(audio.Volume);
volumeMeterEffect.Bindable.BindTo(audio.VolumeSample);
volumeMeterMusic.Bindable.BindTo(audio.VolumeTrack);
2019-11-12 17:45:42 +08:00
IsMuted.BindValueChanged(muted =>
2018-04-13 17:19:50 +08:00
{
if (muted.NewValue)
2018-04-13 17:19:50 +08:00
audio.AddAdjustment(AdjustableProperty.Volume, muteAdjustment);
else
audio.RemoveAdjustment(AdjustableProperty.Volume, muteAdjustment);
2019-10-03 18:11:50 +08:00
});
2018-04-13 17:19:50 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
foreach (var volumeMeter in volumeMeters)
volumeMeter.Bindable.ValueChanged += _ => Show();
muteButton.Current.ValueChanged += _ => Show();
2018-04-13 17:19:50 +08:00
}
public bool Adjust(GlobalAction action, float amount = 1, bool isPrecise = false)
2018-04-13 17:19:50 +08:00
{
if (!IsLoaded) return false;
2018-04-13 17:19:50 +08:00
switch (action)
{
case GlobalAction.DecreaseVolume:
if (State.Value == Visibility.Hidden)
2018-04-13 17:19:50 +08:00
Show();
else
volumeMeters.Selected?.Decrease(amount, isPrecise);
2018-04-13 17:19:50 +08:00
return true;
2019-04-01 11:44:46 +08:00
2018-04-13 17:19:50 +08:00
case GlobalAction.IncreaseVolume:
if (State.Value == Visibility.Hidden)
2018-04-13 17:19:50 +08:00
Show();
else
volumeMeters.Selected?.Increase(amount, isPrecise);
2018-04-13 17:19:50 +08:00
return true;
2019-04-01 11:44:46 +08:00
case GlobalAction.NextVolumeMeter:
if (State.Value == Visibility.Visible)
volumeMeters.SelectNext();
Show();
return true;
case GlobalAction.PreviousVolumeMeter:
if (State.Value == Visibility.Visible)
volumeMeters.SelectPrevious();
Show();
return true;
2018-04-13 17:19:50 +08:00
case GlobalAction.ToggleMute:
Show();
2019-02-21 17:56:34 +08:00
muteButton.Current.Value = !muteButton.Current.Value;
2018-04-13 17:19:50 +08:00
return true;
}
return false;
}
private ScheduledDelegate popOutDelegate;
public void FocusMasterVolume()
{
volumeMeters.Select(volumeMeterMaster);
}
public override void Show()
{
// Focus on the master meter as a default if previously hidden
if (State.Value == Visibility.Hidden)
FocusMasterVolume();
if (State.Value == Visibility.Visible)
schedulePopOut();
base.Show();
}
2018-04-13 17:19:50 +08:00
protected override void PopIn()
{
ClearTransforms();
this.FadeIn(100);
schedulePopOut();
}
protected override void PopOut()
{
this.FadeOut(100);
}
2018-10-02 11:02:47 +08:00
protected override bool OnMouseMove(MouseMoveEvent e)
{
// keep the scheduled event correctly timed as long as we have movement.
schedulePopOut();
2018-10-02 11:02:47 +08:00
return base.OnMouseMove(e);
}
protected override bool OnKeyDown(KeyDownEvent e)
{
switch (e.Key)
{
case Key.Left:
Adjust(GlobalAction.PreviousVolumeMeter);
return true;
case Key.Right:
Adjust(GlobalAction.NextVolumeMeter);
return true;
case Key.Down:
Adjust(GlobalAction.DecreaseVolume);
return true;
case Key.Up:
Adjust(GlobalAction.IncreaseVolume);
return true;
}
return base.OnKeyDown(e);
}
2018-10-02 11:02:47 +08:00
protected override bool OnHover(HoverEvent e)
{
2018-06-22 12:32:00 +08:00
schedulePopOut();
return true;
}
2018-10-02 11:02:47 +08:00
protected override void OnHoverLost(HoverLostEvent e)
{
schedulePopOut();
2018-10-02 11:02:47 +08:00
base.OnHoverLost(e);
}
2018-04-13 17:19:50 +08:00
private void schedulePopOut()
{
popOutDelegate?.Cancel();
this.Delay(1000).Schedule(() =>
{
if (!IsHovered)
Hide();
}, out popOutDelegate);
2018-04-13 17:19:50 +08:00
}
}
}