1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-23 23:27:28 +08:00
osu-lazer/osu.Game.Tournament/Screens/SetupScreen.cs

254 lines
8.5 KiB
C#
Raw Normal View History

2019-09-22 01:10:04 +08:00
// 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.
2019-09-23 03:22:50 +08:00
using System;
2019-11-12 10:04:49 +08:00
using System.Collections.Generic;
using System.Drawing;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Configuration;
2019-09-23 03:22:50 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Game.Graphics.UserInterface;
2019-09-24 18:00:26 +08:00
using osu.Game.Graphics.UserInterfaceV2;
2019-09-23 03:45:23 +08:00
using osu.Game.Online.API;
using osu.Game.Overlays;
using osu.Game.Rulesets;
using osu.Game.Tournament.IPC;
2019-09-23 03:22:50 +08:00
using osuTK;
using osuTK.Graphics;
2019-09-22 01:10:04 +08:00
namespace osu.Game.Tournament.Screens
{
2019-09-23 03:49:21 +08:00
public class SetupScreen : TournamentScreen, IProvideVideo
2019-09-22 01:10:04 +08:00
{
2019-09-23 03:45:23 +08:00
private FillFlowContainer fillFlow;
private LoginOverlay loginOverlay;
private ResolutionSelector resolution;
2019-09-23 03:45:23 +08:00
[Resolved]
private MatchIPCInfo ipc { get; set; }
2019-09-23 03:45:23 +08:00
[Resolved]
private IAPIProvider api { get; set; }
[Resolved]
private RulesetStore rulesets { get; set; }
private Bindable<Size> windowSize;
[BackgroundDependencyLoader]
private void load(FrameworkConfigManager frameworkConfig)
{
windowSize = frameworkConfig.GetBindable<Size>(FrameworkSetting.WindowedSize);
2019-09-23 03:45:23 +08:00
InternalChild = fillFlow = new FillFlowContainer
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Padding = new MarginPadding(10),
Spacing = new Vector2(10),
};
api.LocalUser.BindValueChanged(_ => Schedule(reload));
2019-09-23 03:22:50 +08:00
reload();
}
[Resolved]
private Framework.Game game { get; set; }
2019-09-23 03:22:50 +08:00
private void reload()
{
var fileBasedIpc = ipc as FileBasedIPC;
2019-09-23 03:45:23 +08:00
fillFlow.Children = new Drawable[]
2019-09-23 03:22:50 +08:00
{
new ActionableInfo
{
Label = "Current IPC source",
ButtonText = "Refresh",
Action = () =>
{
fileBasedIpc?.LocateStableStorage();
reload();
},
Value = fileBasedIpc?.Storage?.GetFullPath(string.Empty) ?? "Not found",
Failing = fileBasedIpc?.Storage == null,
Description = "The osu!stable installation which is currently being used as a data source. If a source is not found, make sure you have created an empty ipc.txt in your stable cutting-edge installation, and that it is registered as the default osu! install."
2019-09-23 03:45:23 +08:00
},
new ActionableInfo
{
Label = "Current User",
ButtonText = "Change Login",
Action = () =>
{
api.Logout();
if (loginOverlay == null)
{
2019-09-23 03:49:21 +08:00
AddInternal(loginOverlay = new LoginOverlay
2019-09-23 03:45:23 +08:00
{
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
});
}
loginOverlay.State.Value = Visibility.Visible;
},
Value = api?.LocalUser.Value.Username,
Failing = api?.IsLoggedIn != true,
Description = "In order to access the API and display metadata, a login is required."
},
new LabelledDropdown<RulesetInfo>
{
Label = "Ruleset",
Description = "Decides what stats are displayed and which ranks are retrieved for players",
Items = rulesets.AvailableRulesets,
Current = LadderInfo.Ruleset,
},
resolution = new ResolutionSelector
{
Label = "Stream area resolution",
ButtonText = "Set height",
Action = i =>
{
windowSize.Value = new Size((int)(i * aspect_ratio / TournamentSceneManager.STREAM_AREA_WIDTH * TournamentSceneManager.REQUIRED_WIDTH), i);
}
},
2019-09-23 03:22:50 +08:00
};
}
private const float aspect_ratio = 16f / 9f;
protected override void Update()
{
base.Update();
resolution.Value = $"{ScreenSpaceDrawQuad.Width:N0}x{ScreenSpaceDrawQuad.Height:N0}";
}
2019-11-12 10:04:49 +08:00
public class LabelledDropdown<T> : LabelledComponent<OsuDropdown<T>, T>
{
public LabelledDropdown()
: base(true)
{
}
public IEnumerable<T> Items
{
get => Component.Items;
set => Component.Items = value;
}
protected override OsuDropdown<T> CreateComponent() => new OsuDropdown<T>
{
RelativeSizeAxes = Axes.X,
Width = 0.5f,
};
}
private class ActionableInfo : LabelledDrawable<Drawable>
2019-09-23 03:22:50 +08:00
{
private OsuButton button;
2019-09-23 03:22:50 +08:00
public ActionableInfo()
: base(true)
{
}
public string ButtonText
{
set => button.Text = value;
2019-09-23 03:22:50 +08:00
}
public string Value
{
set => valueText.Text = value;
2019-09-23 03:22:50 +08:00
}
public bool Failing
{
set => valueText.Colour = value ? Color4.Red : Color4.White;
2019-09-23 03:22:50 +08:00
}
public Action Action;
private TournamentSpriteText valueText;
protected FillFlowContainer FlowContainer;
2019-09-23 03:22:50 +08:00
protected override Drawable CreateComponent() => new Container
{
2019-09-23 03:22:50 +08:00
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Children = new Drawable[]
{
valueText = new TournamentSpriteText
2019-09-23 03:22:50 +08:00
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
},
FlowContainer = new FillFlowContainer
2019-09-23 03:22:50 +08:00
{
Anchor = Anchor.CentreRight,
Origin = Anchor.CentreRight,
AutoSizeAxes = Axes.Both,
Spacing = new Vector2(10, 0),
Children = new Drawable[]
{
button = new TriangleButton
{
Size = new Vector2(100, 30),
Action = () => Action?.Invoke()
}
}
}
2019-09-23 03:22:50 +08:00
}
};
}
private class ResolutionSelector : ActionableInfo
{
private const int height_min_allowed_value = 480;
private const int height_max_allowed_value = 2160; // 4k
public new Action<int> Action;
private OsuNumberBox numberBox;
protected override Drawable CreateComponent()
{
var drawable = base.CreateComponent();
FlowContainer.Insert(-1, numberBox = new OsuNumberBox
{
Width = 100
});
base.Action = () =>
{
if (numberBox.Text.Length > 0)
{
// box contains text
if (!int.TryParse(numberBox.Text, out var number))
{
// at this point, the only reason we can arrive here is if the input number was too big to parse into an int
// so clamp to max allowed value
number = height_max_allowed_value;
}
else
{
number = Math.Clamp(number, height_min_allowed_value, height_max_allowed_value);
}
// in case number got clamped, reset number in numberBox
numberBox.Text = number.ToString();
Action?.Invoke(number);
}
};
return drawable;
}
}
2019-09-22 01:10:04 +08:00
}
}