1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-12 06:07:28 +08:00
osu-lazer/osu.Game/Overlays/Settings/Sections/Graphics/LayoutSettings.cs

210 lines
8.0 KiB
C#
Raw Normal View History

2018-04-13 17:19:50 +08:00
// Copyright (c) 2007-2018 ppy Pty Ltd <contact@ppy.sh>.
// Licensed under the MIT Licence - https://raw.githubusercontent.com/ppy/osu/master/LICENCE
using System;
using System.Collections.Generic;
2018-06-10 21:17:57 +08:00
using System.Drawing;
using System.Linq;
2018-04-13 17:19:50 +08:00
using osu.Framework.Allocation;
using osu.Framework.Configuration;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Threading;
2019-01-04 12:29:37 +08:00
using osu.Game.Configuration;
2018-11-14 17:02:38 +08:00
using osu.Game.Graphics.UserInterface;
using osuTK.Input;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Overlays.Settings.Sections.Graphics
{
public class LayoutSettings : SettingsSubsection
{
protected override string Header => "Layout";
2019-01-04 12:29:37 +08:00
private FillFlowContainer scalingSettings;
2018-04-13 17:19:50 +08:00
2019-01-04 12:29:37 +08:00
private Bindable<ScalingMode> scalingMode;
2018-06-10 21:17:57 +08:00
private Bindable<Size> sizeFullscreen;
2018-04-13 17:19:50 +08:00
private OsuGameBase game;
2018-09-05 10:32:38 +08:00
private SettingsDropdown<Size> resolutionDropdown;
private SettingsEnumDropdown<WindowMode> windowModeDropdown;
private Bindable<float> scalingPositionX;
private Bindable<float> scalingPositionY;
private Bindable<float> scalingSizeX;
private Bindable<float> scalingSizeY;
2018-04-13 17:19:50 +08:00
private const int transition_duration = 400;
[BackgroundDependencyLoader]
2019-01-04 12:29:37 +08:00
private void load(FrameworkConfigManager config, OsuConfigManager osuConfig, OsuGameBase game)
2018-04-13 17:19:50 +08:00
{
this.game = game;
2019-01-04 12:29:37 +08:00
scalingMode = osuConfig.GetBindable<ScalingMode>(OsuSetting.Scaling);
2018-06-10 21:17:57 +08:00
sizeFullscreen = config.GetBindable<Size>(FrameworkSetting.SizeFullscreen);
scalingSizeX = osuConfig.GetBindable<float>(OsuSetting.ScalingSizeX);
scalingSizeY = osuConfig.GetBindable<float>(OsuSetting.ScalingSizeY);
scalingPositionX = osuConfig.GetBindable<float>(OsuSetting.ScalingPositionX);
scalingPositionY = osuConfig.GetBindable<float>(OsuSetting.ScalingPositionY);
2018-06-10 21:17:57 +08:00
2018-09-19 16:52:35 +08:00
Container resolutionSettingsContainer;
2018-04-13 17:19:50 +08:00
Children = new Drawable[]
{
windowModeDropdown = new SettingsEnumDropdown<WindowMode>
2018-04-13 17:19:50 +08:00
{
LabelText = "Screen mode",
Bindable = config.GetBindable<WindowMode>(FrameworkSetting.WindowMode),
},
2018-09-19 16:52:35 +08:00
resolutionSettingsContainer = new Container
{
2018-09-19 16:52:35 +08:00
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y
},
2019-01-04 12:29:37 +08:00
new SettingsEnumDropdown<ScalingMode>
2018-04-13 17:19:50 +08:00
{
2019-01-04 12:29:37 +08:00
LabelText = "Scaling",
Bindable = osuConfig.GetBindable<ScalingMode>(OsuSetting.Scaling),
2018-04-13 17:19:50 +08:00
},
2019-01-04 12:29:37 +08:00
scalingSettings = new FillFlowContainer
2018-04-13 17:19:50 +08:00
{
Direction = FillDirection.Vertical,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
AutoSizeDuration = transition_duration,
AutoSizeEasing = Easing.OutQuint,
Masking = true,
Children = new Drawable[]
{
2019-01-04 12:29:37 +08:00
new SettingsSlider<float>
2018-04-13 17:19:50 +08:00
{
LabelText = "Horizontal position",
Bindable = delayedBindable(scalingPositionX),
2018-07-10 23:28:56 +08:00
KeyboardStep = 0.01f
2018-04-13 17:19:50 +08:00
},
2019-01-04 12:29:37 +08:00
new SettingsSlider<float>
2018-04-13 17:19:50 +08:00
{
LabelText = "Vertical position",
Bindable = delayedBindable(scalingPositionY),
2019-01-04 12:29:37 +08:00
KeyboardStep = 0.01f
},
new SettingsSlider<float>
{
2019-01-04 13:58:44 +08:00
LabelText = "Horizontal scale",
Bindable = delayedBindable(scalingSizeX),
2019-01-04 12:29:37 +08:00
KeyboardStep = 0.01f
},
new SettingsSlider<float>
{
2019-01-04 13:58:44 +08:00
LabelText = "Vertical scale",
Bindable = delayedBindable(scalingSizeY),
2018-07-10 23:28:56 +08:00
KeyboardStep = 0.01f
2018-04-13 17:19:50 +08:00
},
}
},
};
2018-09-19 16:52:35 +08:00
var resolutions = getResolutions();
2018-09-20 18:00:41 +08:00
2018-09-19 16:52:35 +08:00
if (resolutions.Count > 1)
{
2018-11-14 17:02:38 +08:00
resolutionSettingsContainer.Child = resolutionDropdown = new ResolutionSettingsDropdown
2018-06-22 01:42:22 +08:00
{
2018-09-19 16:52:35 +08:00
LabelText = "Resolution",
ShowsDefaultIndicator = false,
2018-09-20 18:00:41 +08:00
Items = resolutions,
2018-09-19 16:52:35 +08:00
Bindable = sizeFullscreen
};
windowModeDropdown.Bindable.BindValueChanged(windowMode =>
{
if (windowMode == WindowMode.Fullscreen)
{
resolutionDropdown.Show();
sizeFullscreen.TriggerChange();
}
else
resolutionDropdown.Hide();
}, true);
2018-09-19 16:52:35 +08:00
}
2019-01-04 12:29:37 +08:00
scalingMode.BindValueChanged(mode =>
2018-04-13 17:19:50 +08:00
{
2019-01-04 12:29:37 +08:00
scalingSettings.ClearTransforms();
scalingSettings.AutoSizeAxes = mode != ScalingMode.Off ? Axes.Y : Axes.None;
2018-04-13 17:19:50 +08:00
2019-01-04 12:29:37 +08:00
if (mode == ScalingMode.Off)
scalingSettings.ResizeHeightTo(0, transition_duration, Easing.OutQuint);
2018-09-05 10:32:38 +08:00
}, true);
2018-04-13 17:19:50 +08:00
}
/// <summary>
/// Create a delayed bindable which only updates when a condition is met.
/// </summary>
/// <param name="configBindable">The config bindable.</param>
/// <returns>A bindable which will propagate updates with a delay.</returns>
private Bindable<float> delayedBindable(Bindable<float> configBindable)
{
var delayed = new BindableFloat { MinValue = 0, MaxValue = 1, Default = configBindable.Default };
configBindable.BindValueChanged(v => delayed.Value = v, true);
delayed.ValueChanged += v =>
{
if (scalingMode == ScalingMode.Everything)
applyWithDelay(() => configBindable.Value = v);
else
configBindable.Value = v;
};
return delayed;
}
private ScheduledDelegate delayedApplication;
private void applyWithDelay(Action func, bool firstRun = true)
{
if (!firstRun && !GetContainingInputManager().CurrentState.Mouse.IsPressed(MouseButton.Left))
{
func();
return;
}
delayedApplication?.Cancel();
delayedApplication = Scheduler.AddDelayed(() => applyWithDelay(func, false), 250);
}
2018-11-14 17:02:38 +08:00
private IReadOnlyList<Size> getResolutions()
{
2018-11-14 17:02:38 +08:00
var resolutions = new List<Size> { new Size(9999, 9999) };
if (game.Window != null)
2018-11-14 17:02:38 +08:00
{
resolutions.AddRange(game.Window.AvailableResolutions
.Where(r => r.Width >= 800 && r.Height >= 600)
.OrderByDescending(r => r.Width)
.ThenByDescending(r => r.Height)
.Select(res => new Size(res.Width, res.Height))
.Distinct());
}
return resolutions;
}
private class ResolutionSettingsDropdown : SettingsDropdown<Size>
{
protected override OsuDropdown<Size> CreateDropdown() => new ResolutionDropdownControl { Items = Items };
private class ResolutionDropdownControl : DropdownControl
{
protected override string GenerateItemText(Size item)
{
if (item == new Size(9999, 9999))
return "Default";
return $"{item.Width}x{item.Height}";
}
}
}
2018-04-13 17:19:50 +08:00
}
}