1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-20 04:42:56 +08:00
osu-lazer/osu.Game/Screens/OnlinePlay/Multiplayer/Match/MultiplayerMatchSettingsOverlay.cs

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

585 lines
29 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.
using System;
using System.ComponentModel;
2020-12-29 15:20:43 +08:00
using System.Diagnostics;
2024-11-14 16:57:32 +08:00
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions;
using osu.Framework.Extensions.ExceptionExtensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Screens;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.Sprites;
using osu.Game.Graphics.UserInterface;
using osu.Game.Graphics.UserInterfaceV2;
using osu.Game.Online.Multiplayer;
2020-12-25 12:38:11 +08:00
using osu.Game.Online.Rooms;
using osu.Game.Overlays;
using osu.Game.Screens.OnlinePlay.Match.Components;
using osuTK;
using Container = osu.Framework.Graphics.Containers.Container;
namespace osu.Game.Screens.OnlinePlay.Multiplayer.Match
{
public partial class MultiplayerMatchSettingsOverlay : RoomSettingsOverlay
{
public required Bindable<PlaylistItem?> SelectedItem
{
get => selectedItem;
set => selectedItem.Current = value;
}
protected override OsuButton SubmitButton => settings.ApplyButton;
protected override bool IsLoading => ongoingOperationTracker.InProgress.Value;
[Resolved]
2022-09-07 18:05:53 +08:00
private OngoingOperationTracker ongoingOperationTracker { get; set; } = null!;
private readonly BindableWithCurrent<PlaylistItem?> selectedItem = new BindableWithCurrent<PlaylistItem?>();
private MatchSettings settings = null!;
2021-08-19 15:40:27 +08:00
public MultiplayerMatchSettingsOverlay(Room room)
: base(room)
{
}
protected override void SelectBeatmap() => settings.SelectBeatmap();
2021-08-19 15:40:27 +08:00
protected override OnlinePlayComposite CreateSettings(Room room) => settings = new MatchSettings(room)
{
RelativeSizeAxes = Axes.Both,
RelativePositionAxes = Axes.Y,
SettingsApplied = Hide,
SelectedItem = { BindTarget = SelectedItem }
2021-08-19 15:40:27 +08:00
};
protected partial class MatchSettings : OnlinePlayComposite
{
private const float disabled_alpha = 0.2f;
public override bool IsPresent => base.IsPresent || Scheduler.HasPendingTasks;
public readonly Bindable<PlaylistItem?> SelectedItem = new Bindable<PlaylistItem?>();
2022-09-07 18:05:53 +08:00
public Action? SettingsApplied;
public OsuTextBox NameField = null!;
public OsuTextBox MaxParticipantsField = null!;
public MatchTypePicker TypePicker = null!;
public OsuEnumDropdown<QueueMode> QueueModeDropdown = null!;
public OsuTextBox PasswordTextBox = null!;
public OsuCheckbox AutoSkipCheckbox = null!;
public RoundedButton ApplyButton = null!;
2022-09-07 18:05:53 +08:00
public OsuSpriteText ErrorText = null!;
2022-09-07 18:05:53 +08:00
private OsuEnumDropdown<StartMode> startModeDropdown = null!;
private OsuSpriteText typeLabel = null!;
private LoadingLayer loadingLayer = null!;
public void SelectBeatmap() => selectBeatmapButton.TriggerClick();
[Resolved]
2022-09-07 18:05:53 +08:00
private MultiplayerMatchSubScreen matchSubScreen { get; set; } = null!;
[Resolved]
2022-09-07 18:05:53 +08:00
private IRoomManager manager { get; set; } = null!;
[Resolved]
2022-09-07 18:05:53 +08:00
private MultiplayerClient client { get; set; } = null!;
2020-12-29 15:20:43 +08:00
[Resolved]
2022-09-07 18:05:53 +08:00
private OngoingOperationTracker ongoingOperationTracker { get; set; } = null!;
2020-12-29 15:20:43 +08:00
private readonly IBindable<bool> operationInProgress = new BindableBool();
2021-08-19 15:40:27 +08:00
private readonly Room room;
2022-09-07 18:05:53 +08:00
private IDisposable? applyingSettingsOperation;
private Drawable playlistContainer = null!;
private DrawableRoomPlaylist drawablePlaylist = null!;
private RoundedButton selectBeatmapButton = null!;
2021-08-19 15:40:27 +08:00
public MatchSettings(Room room)
{
this.room = room;
}
[BackgroundDependencyLoader]
private void load(OverlayColourProvider colourProvider, OsuColour colours)
{
InternalChildren = new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colourProvider.Background4
},
new GridContainer
{
RelativeSizeAxes = Axes.Both,
RowDimensions = new[]
{
2021-07-05 23:52:39 +08:00
new Dimension(),
new Dimension(GridSizeMode.AutoSize),
},
Content = new[]
{
new Drawable[]
{
new OsuScrollContainer
{
Padding = new MarginPadding
{
Horizontal = OsuScreen.HORIZONTAL_OVERFLOW_PADDING,
Vertical = 10
},
RelativeSizeAxes = Axes.Both,
Children = new[]
{
new FillFlowContainer
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(0, 10),
Children = new[]
{
new Container
{
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
Padding = new MarginPadding { Horizontal = WaveOverlayContainer.WIDTH_PADDING },
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Children = new Drawable[]
{
new SectionContainer
{
Padding = new MarginPadding { Right = FIELD_PADDING / 2 },
Children = new[]
{
new Section("Room name")
{
Child = NameField = new OsuTextBox
{
RelativeSizeAxes = Axes.X,
TabbableContentContainer = this,
LengthLimit = 100,
},
},
// new Section("Room visibility")
// {
// Alpha = disabled_alpha,
// Child = AvailabilityPicker = new RoomAvailabilityPicker
// {
// Enabled = { Value = false }
// },
// },
new Section("Game type")
{
Child = new FillFlowContainer
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
Direction = FillDirection.Vertical,
Spacing = new Vector2(7),
Children = new Drawable[]
{
TypePicker = new MatchTypePicker
{
RelativeSizeAxes = Axes.X,
},
typeLabel = new OsuSpriteText
{
Font = OsuFont.GetFont(size: 14),
Colour = colours.Yellow
},
},
},
},
2021-11-19 14:47:38 +08:00
new Section("Queue mode")
2021-10-20 13:51:59 +08:00
{
Child = new Container
2021-10-20 13:51:59 +08:00
{
RelativeSizeAxes = Axes.X,
Height = 40,
2021-11-16 13:53:10 +08:00
Child = QueueModeDropdown = new OsuEnumDropdown<QueueMode>
{
RelativeSizeAxes = Axes.X
}
2021-10-20 13:51:59 +08:00
}
},
new Section("Auto start")
{
Child = new Container
{
RelativeSizeAxes = Axes.X,
Height = 40,
Child = startModeDropdown = new OsuEnumDropdown<StartMode>
{
RelativeSizeAxes = Axes.X
}
}
2021-10-20 13:51:59 +08:00
}
},
},
new SectionContainer
{
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
Padding = new MarginPadding { Left = FIELD_PADDING / 2 },
Children = new[]
{
new Section("Max participants")
{
Alpha = disabled_alpha,
Child = MaxParticipantsField = new OsuNumberBox
{
RelativeSizeAxes = Axes.X,
TabbableContentContainer = this,
ReadOnly = true,
},
},
new Section("Password (optional)")
{
Child = PasswordTextBox = new OsuPasswordTextBox
{
RelativeSizeAxes = Axes.X,
TabbableContentContainer = this,
LengthLimit = 255,
},
},
new Section("Other")
{
Child = AutoSkipCheckbox = new OsuCheckbox
{
LabelText = "Automatically skip the beatmap intro"
}
}
}
}
},
},
playlistContainer = new FillFlowContainer
{
Anchor = Anchor.TopCentre,
Origin = Anchor.TopCentre,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Width = 0.5f,
Depth = float.MaxValue,
Spacing = new Vector2(5),
Children = new Drawable[]
{
drawablePlaylist = new DrawableRoomPlaylist
{
RelativeSizeAxes = Axes.X,
2024-11-14 16:57:32 +08:00
Height = DrawableRoomPlaylistItem.HEIGHT,
SelectedItem = { BindTarget = SelectedItem }
},
selectBeatmapButton = new RoundedButton
{
RelativeSizeAxes = Axes.X,
Height = 40,
Text = "Select beatmap",
Action = () =>
{
if (matchSubScreen.IsCurrentScreen())
matchSubScreen.Push(new MultiplayerMatchSongSelect(matchSubScreen.Room));
}
}
}
}
}
}
},
},
},
new Drawable[]
{
new Container
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
Y = 2,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Children = new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colourProvider.Background5
},
new FillFlowContainer
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Direction = FillDirection.Vertical,
Spacing = new Vector2(0, 20),
Margin = new MarginPadding { Vertical = 20 },
Padding = new MarginPadding { Horizontal = OsuScreen.HORIZONTAL_OVERFLOW_PADDING },
Children = new Drawable[]
{
ApplyButton = new CreateOrUpdateButton(room)
{
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
Size = new Vector2(230, 55),
Enabled = { Value = false },
Action = apply,
},
ErrorText = new OsuSpriteText
{
Anchor = Anchor.BottomCentre,
Origin = Anchor.BottomCentre,
Alpha = 0,
Depth = 1,
Colour = colours.RedDark
}
}
}
}
}
}
}
},
loadingLayer = new LoadingLayer(true)
};
TypePicker.Current.BindValueChanged(type => typeLabel.Text = type.NewValue.GetLocalisableDescription(), true);
2020-12-29 15:20:43 +08:00
operationInProgress.BindTo(ongoingOperationTracker.InProgress);
operationInProgress.BindValueChanged(v =>
{
if (v.NewValue)
loadingLayer.Show();
else
loadingLayer.Hide();
});
}
protected override void LoadComplete()
{
base.LoadComplete();
2024-11-13 15:55:18 +08:00
room.PropertyChanged += onRoomPropertyChanged;
updateRoomName();
2024-11-13 17:36:47 +08:00
updateRoomType();
2024-11-13 17:57:51 +08:00
updateRoomQueueMode();
updateRoomPassword();
2024-11-13 18:15:29 +08:00
updateRoomAutoSkip();
updateRoomMaxParticipants();
updateRoomAutoStartDuration();
2024-11-14 16:57:32 +08:00
updateRoomPlaylist();
drawablePlaylist.Items.BindCollectionChanged((_, __) => room.Playlist = drawablePlaylist.Items.ToArray());
2024-11-13 15:55:18 +08:00
}
private void onRoomPropertyChanged(object? sender, PropertyChangedEventArgs e)
{
2024-11-13 17:36:47 +08:00
switch (e.PropertyName)
{
case nameof(Room.Name):
updateRoomName();
break;
case nameof(Room.Type):
updateRoomName();
break;
2024-11-13 17:57:51 +08:00
case nameof(Room.QueueMode):
updateRoomQueueMode();
break;
case nameof(Room.Password):
updateRoomPassword();
break;
2024-11-13 18:15:29 +08:00
case nameof(Room.AutoSkip):
updateRoomAutoSkip();
break;
case nameof(Room.MaxParticipants):
updateRoomMaxParticipants();
break;
case nameof(Room.AutoStartDuration):
updateRoomAutoStartDuration();
break;
2024-11-14 16:57:32 +08:00
case nameof(Room.Playlist):
updateRoomPlaylist();
break;
2024-11-13 17:36:47 +08:00
}
}
2024-11-13 15:55:18 +08:00
private void updateRoomName()
=> NameField.Text = room.Name;
2024-11-13 17:36:47 +08:00
private void updateRoomType()
=> TypePicker.Current.Value = room.Type;
2024-11-13 17:57:51 +08:00
private void updateRoomQueueMode()
=> QueueModeDropdown.Current.Value = room.QueueMode;
private void updateRoomPassword()
=> PasswordTextBox.Text = room.Password ?? string.Empty;
2024-11-13 18:15:29 +08:00
private void updateRoomAutoSkip()
=> AutoSkipCheckbox.Current.Value = room.AutoSkip;
private void updateRoomMaxParticipants()
=> MaxParticipantsField.Text = room.MaxParticipants?.ToString();
private void updateRoomAutoStartDuration()
=> typeLabel.Text = room.AutoStartDuration.GetLocalisableDescription();
2024-11-14 16:57:32 +08:00
private void updateRoomPlaylist()
=> drawablePlaylist.Items.ReplaceRange(0, drawablePlaylist.Items.Count, room.Playlist);
protected override void Update()
{
base.Update();
2024-11-14 16:57:32 +08:00
ApplyButton.Enabled.Value = room.Playlist.Count > 0 && NameField.Text.Length > 0 && !operationInProgress.Value;
playlistContainer.Alpha = room.RoomID == null ? 1 : 0;
}
private void apply()
{
if (!ApplyButton.Enabled.Value)
return;
hideError();
2020-12-29 15:20:43 +08:00
Debug.Assert(applyingSettingsOperation == null);
applyingSettingsOperation = ongoingOperationTracker.BeginOperation();
// If the client is already in a room, update via the client.
// Otherwise, update the room directly in preparation for it to be submitted to the API on match creation.
if (client.Room != null)
{
2021-10-20 19:25:06 +08:00
client.ChangeSettings(
name: NameField.Text,
password: PasswordTextBox.Text,
matchType: TypePicker.Current.Value,
queueMode: QueueModeDropdown.Current.Value,
autoStartDuration: TimeSpan.FromSeconds((int)startModeDropdown.Current.Value),
autoSkip: AutoSkipCheckbox.Current.Value)
2021-10-20 19:25:06 +08:00
.ContinueWith(t => Schedule(() =>
{
if (t.IsCompletedSuccessfully)
onSuccess(room);
else
onError(t.Exception?.AsSingular().Message ?? "Error changing settings.");
}));
}
else
{
2024-11-13 15:55:18 +08:00
room.Name = NameField.Text;
2024-11-13 17:36:47 +08:00
room.Type = TypePicker.Current.Value;
room.Password = PasswordTextBox.Current.Value;
2024-11-13 17:57:51 +08:00
room.QueueMode = QueueModeDropdown.Current.Value;
room.AutoStartDuration = TimeSpan.FromSeconds((int)startModeDropdown.Current.Value);
2024-11-13 18:15:29 +08:00
room.AutoSkip = AutoSkipCheckbox.Current.Value;
if (int.TryParse(MaxParticipantsField.Text, out int max))
room.MaxParticipants = max;
else
room.MaxParticipants = null;
2022-09-07 18:05:53 +08:00
manager.CreateRoom(room, onSuccess, onError);
}
}
private void hideError() => ErrorText.FadeOut(50);
private void onSuccess(Room room) => Schedule(() =>
{
2020-12-29 15:20:43 +08:00
Debug.Assert(applyingSettingsOperation != null);
SettingsApplied?.Invoke();
2020-12-29 15:20:43 +08:00
applyingSettingsOperation.Dispose();
applyingSettingsOperation = null;
});
private void onError(string text) => Schedule(() =>
{
2020-12-29 15:20:43 +08:00
Debug.Assert(applyingSettingsOperation != null);
// see https://github.com/ppy/osu-web/blob/2c97aaeb64fb4ed97c747d8383a35b30f57428c7/app/Models/Multiplayer/PlaylistItem.php#L48.
const string not_found_prefix = "beatmaps not found:";
if (text.StartsWith(not_found_prefix, StringComparison.Ordinal))
{
ErrorText.Text = "The selected beatmap is not available online.";
SelectedItem.Value?.MarkInvalid();
}
else
{
ErrorText.Text = text;
}
ErrorText.FadeIn(50);
2020-12-29 15:20:43 +08:00
applyingSettingsOperation.Dispose();
applyingSettingsOperation = null;
});
2024-11-13 15:55:18 +08:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
room.PropertyChanged -= onRoomPropertyChanged;
}
}
public partial class CreateOrUpdateButton : RoundedButton
{
private readonly Room room;
public CreateOrUpdateButton(Room room)
{
this.room = room;
}
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
BackgroundColour = colours.YellowDark;
}
protected override void Update()
{
base.Update();
Text = room.RoomID == null ? "Create" : "Update";
}
}
private enum StartMode
{
[Description("Off")]
Off = 0,
[Description("30 seconds")]
Seconds30 = 30,
[Description("1 minute")]
Seconds60 = 60,
[Description("3 minutes")]
Seconds180 = 180,
[Description("5 minutes")]
Seconds300 = 300
}
}
}