2021-02-01 13:57:39 +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.
|
|
|
|
|
2022-06-17 15:37:17 +08:00
|
|
|
#nullable disable
|
|
|
|
|
2021-02-01 13:57:39 +08:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
|
|
|
using Humanizer;
|
2021-02-17 18:06:37 +08:00
|
|
|
using JetBrains.Annotations;
|
2021-02-01 13:57:39 +08:00
|
|
|
using osu.Framework.Allocation;
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
using osu.Framework.Screens;
|
|
|
|
using osu.Game.Beatmaps;
|
2022-02-15 15:01:14 +08:00
|
|
|
using osu.Game.Online.API;
|
2021-02-01 13:57:39 +08:00
|
|
|
using osu.Game.Online.Rooms;
|
2022-05-05 04:17:40 +08:00
|
|
|
using osu.Game.Overlays;
|
2021-02-01 13:57:39 +08:00
|
|
|
using osu.Game.Overlays.Mods;
|
|
|
|
using osu.Game.Rulesets;
|
|
|
|
using osu.Game.Rulesets.Mods;
|
|
|
|
using osu.Game.Screens.Select;
|
2021-08-14 22:39:12 +08:00
|
|
|
using osu.Game.Users;
|
2021-02-02 11:49:49 +08:00
|
|
|
using osu.Game.Utils;
|
2021-02-01 13:57:39 +08:00
|
|
|
|
|
|
|
namespace osu.Game.Screens.OnlinePlay
|
|
|
|
{
|
|
|
|
public abstract class OnlinePlaySongSelect : SongSelect, IOnlinePlaySubScreen
|
|
|
|
{
|
|
|
|
public string ShortTitle => "song selection";
|
|
|
|
|
|
|
|
public override string Title => ShortTitle.Humanize();
|
|
|
|
|
|
|
|
public override bool AllowEditing => false;
|
|
|
|
|
|
|
|
[Resolved(typeof(Room), nameof(Room.Playlist))]
|
|
|
|
protected BindableList<PlaylistItem> Playlist { get; private set; }
|
|
|
|
|
2021-12-10 00:15:15 +08:00
|
|
|
[CanBeNull]
|
|
|
|
[Resolved(CanBeNull = true)]
|
|
|
|
protected IBindable<PlaylistItem> SelectedItem { get; private set; }
|
|
|
|
|
2022-02-15 15:01:14 +08:00
|
|
|
[Resolved]
|
|
|
|
private RulesetStore rulesets { get; set; }
|
|
|
|
|
2021-08-14 22:39:12 +08:00
|
|
|
protected override UserActivity InitialActivity => new UserActivity.InLobby(room);
|
|
|
|
|
2021-02-16 14:14:19 +08:00
|
|
|
protected readonly Bindable<IReadOnlyList<Mod>> FreeMods = new Bindable<IReadOnlyList<Mod>>(Array.Empty<Mod>());
|
|
|
|
|
2021-08-24 12:29:19 +08:00
|
|
|
private readonly Room room;
|
2021-02-01 13:57:39 +08:00
|
|
|
|
|
|
|
private WorkingBeatmap initialBeatmap;
|
|
|
|
private RulesetInfo initialRuleset;
|
|
|
|
private IReadOnlyList<Mod> initialMods;
|
|
|
|
private bool itemSelected;
|
|
|
|
|
2022-05-11 04:29:57 +08:00
|
|
|
private readonly FreeModSelectOverlay freeModSelectOverlay;
|
2022-05-05 04:25:13 +08:00
|
|
|
private IDisposable freeModSelectOverlayRegistration;
|
|
|
|
|
2021-08-24 12:29:19 +08:00
|
|
|
protected OnlinePlaySongSelect(Room room)
|
2021-02-01 13:57:39 +08:00
|
|
|
{
|
2021-08-24 12:29:19 +08:00
|
|
|
this.room = room;
|
|
|
|
|
2021-02-01 13:57:39 +08:00
|
|
|
Padding = new MarginPadding { Horizontal = HORIZONTAL_OVERFLOW_PADDING };
|
|
|
|
|
2022-05-11 04:29:57 +08:00
|
|
|
freeModSelectOverlay = new FreeModSelectOverlay
|
2021-02-01 13:57:39 +08:00
|
|
|
{
|
2021-02-16 14:14:19 +08:00
|
|
|
SelectedMods = { BindTarget = FreeMods },
|
2021-02-01 13:57:39 +08:00
|
|
|
IsValidMod = IsValidFreeMod,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
private void load()
|
|
|
|
{
|
2021-08-12 17:02:00 +08:00
|
|
|
LeftArea.Padding = new MarginPadding { Top = Header.HEIGHT };
|
|
|
|
|
2021-02-01 13:57:39 +08:00
|
|
|
initialBeatmap = Beatmap.Value;
|
|
|
|
initialRuleset = Ruleset.Value;
|
|
|
|
initialMods = Mods.Value.ToList();
|
|
|
|
|
2022-05-05 04:25:13 +08:00
|
|
|
LoadComponent(freeModSelectOverlay);
|
2021-02-02 17:56:57 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected override void LoadComplete()
|
|
|
|
{
|
|
|
|
base.LoadComplete();
|
|
|
|
|
2022-02-15 15:01:14 +08:00
|
|
|
var rulesetInstance = SelectedItem?.Value?.RulesetID == null ? null : rulesets.GetRuleset(SelectedItem.Value.RulesetID)?.CreateInstance();
|
|
|
|
|
|
|
|
if (rulesetInstance != null)
|
|
|
|
{
|
|
|
|
// At this point, Mods contains both the required and allowed mods. For selection purposes, it should only contain the required mods.
|
|
|
|
// Similarly, freeMods is currently empty but should only contain the allowed mods.
|
|
|
|
Mods.Value = SelectedItem.Value.RequiredMods.Select(m => m.ToMod(rulesetInstance)).ToArray();
|
|
|
|
FreeMods.Value = SelectedItem.Value.AllowedMods.Select(m => m.ToMod(rulesetInstance)).ToArray();
|
|
|
|
}
|
2021-02-02 12:54:27 +08:00
|
|
|
|
2021-02-22 12:21:50 +08:00
|
|
|
Mods.BindValueChanged(onModsChanged);
|
2021-02-02 12:54:27 +08:00
|
|
|
Ruleset.BindValueChanged(onRulesetChanged);
|
2022-05-05 04:25:13 +08:00
|
|
|
|
2022-05-05 21:47:10 +08:00
|
|
|
freeModSelectOverlayRegistration = OverlayManager?.RegisterBlockingOverlay(freeModSelectOverlay);
|
2021-02-02 12:54:27 +08:00
|
|
|
}
|
|
|
|
|
2021-02-22 12:21:50 +08:00
|
|
|
private void onModsChanged(ValueChangedEvent<IReadOnlyList<Mod>> mods)
|
|
|
|
{
|
|
|
|
FreeMods.Value = FreeMods.Value.Where(checkCompatibleFreeMod).ToList();
|
|
|
|
|
|
|
|
// Reset the validity delegate to update the overlay's display.
|
|
|
|
freeModSelectOverlay.IsValidMod = IsValidFreeMod;
|
|
|
|
}
|
|
|
|
|
2021-02-02 12:54:27 +08:00
|
|
|
private void onRulesetChanged(ValueChangedEvent<RulesetInfo> ruleset)
|
|
|
|
{
|
2021-02-16 14:14:19 +08:00
|
|
|
FreeMods.Value = Array.Empty<Mod>();
|
2021-02-01 13:57:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected sealed override bool OnStart()
|
|
|
|
{
|
2022-02-15 22:33:26 +08:00
|
|
|
var item = new PlaylistItem(Beatmap.Value.BeatmapInfo)
|
2021-06-01 13:33:21 +08:00
|
|
|
{
|
2022-02-15 15:01:14 +08:00
|
|
|
RulesetID = Ruleset.Value.OnlineID,
|
|
|
|
RequiredMods = Mods.Value.Select(m => new APIMod(m)).ToArray(),
|
|
|
|
AllowedMods = FreeMods.Value.Select(m => new APIMod(m)).ToArray()
|
2021-06-01 13:33:21 +08:00
|
|
|
};
|
2021-02-01 13:57:39 +08:00
|
|
|
|
2022-06-18 13:28:25 +08:00
|
|
|
if (SelectItem(item))
|
|
|
|
{
|
|
|
|
itemSelected = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-02-01 13:57:39 +08:00
|
|
|
}
|
|
|
|
|
2021-02-01 17:50:32 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Invoked when the user has requested a selection of a beatmap.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="item">The resultant <see cref="PlaylistItem"/>. This item has not yet been added to the <see cref="Room"/>'s.</param>
|
2022-06-18 13:28:25 +08:00
|
|
|
/// <returns><c>true</c> if a selection occurred.</returns>
|
|
|
|
protected abstract bool SelectItem(PlaylistItem item);
|
2021-02-01 13:57:39 +08:00
|
|
|
|
|
|
|
public override bool OnBackButton()
|
|
|
|
{
|
|
|
|
if (freeModSelectOverlay.State.Value == Visibility.Visible)
|
|
|
|
{
|
|
|
|
freeModSelectOverlay.Hide();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return base.OnBackButton();
|
|
|
|
}
|
|
|
|
|
2022-04-21 23:52:44 +08:00
|
|
|
public override bool OnExiting(ScreenExitEvent e)
|
2021-02-01 13:57:39 +08:00
|
|
|
{
|
|
|
|
if (!itemSelected)
|
|
|
|
{
|
|
|
|
Beatmap.Value = initialBeatmap;
|
|
|
|
Ruleset.Value = initialRuleset;
|
|
|
|
Mods.Value = initialMods;
|
|
|
|
}
|
|
|
|
|
2022-05-05 23:27:21 +08:00
|
|
|
freeModSelectOverlay.Hide();
|
|
|
|
|
2022-04-21 23:52:44 +08:00
|
|
|
return base.OnExiting(e);
|
2021-02-01 13:57:39 +08:00
|
|
|
}
|
|
|
|
|
2022-05-11 04:29:57 +08:00
|
|
|
protected override ModSelectOverlay CreateModSelectOverlay() => new UserModSelectOverlay(OverlayColourScheme.Plum)
|
2021-02-01 13:57:39 +08:00
|
|
|
{
|
|
|
|
IsValidMod = IsValidMod
|
|
|
|
};
|
|
|
|
|
|
|
|
protected override IEnumerable<(FooterButton, OverlayContainer)> CreateFooterButtons()
|
|
|
|
{
|
|
|
|
var buttons = base.CreateFooterButtons().ToList();
|
2021-02-16 14:14:19 +08:00
|
|
|
buttons.Insert(buttons.FindIndex(b => b.Item1 is FooterButtonMods) + 1, (new FooterButtonFreeMods { Current = FreeMods }, freeModSelectOverlay));
|
2021-02-01 13:57:39 +08:00
|
|
|
return buttons;
|
|
|
|
}
|
|
|
|
|
2021-02-01 17:50:32 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Checks whether a given <see cref="Mod"/> is valid for global selection.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="mod">The <see cref="Mod"/> to check.</param>
|
|
|
|
/// <returns>Whether <paramref name="mod"/> is a valid mod for online play.</returns>
|
2021-06-09 13:32:48 +08:00
|
|
|
protected virtual bool IsValidMod(Mod mod) => mod.HasImplementation && ModUtils.FlattenMod(mod).All(m => m.UserPlayable);
|
2021-02-01 13:57:39 +08:00
|
|
|
|
2021-02-01 17:50:32 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Checks whether a given <see cref="Mod"/> is valid for per-player free-mod selection.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="mod">The <see cref="Mod"/> to check.</param>
|
|
|
|
/// <returns>Whether <paramref name="mod"/> is a selectable free-mod.</returns>
|
2021-02-22 12:21:50 +08:00
|
|
|
protected virtual bool IsValidFreeMod(Mod mod) => IsValidMod(mod) && checkCompatibleFreeMod(mod);
|
|
|
|
|
|
|
|
private bool checkCompatibleFreeMod(Mod mod)
|
|
|
|
=> Mods.Value.All(m => m.Acronym != mod.Acronym) // Mod must not be contained in the required mods.
|
|
|
|
&& ModUtils.CheckCompatibleSet(Mods.Value.Append(mod).ToArray()); // Mod must be compatible with all the required mods.
|
2022-05-05 04:25:13 +08:00
|
|
|
|
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
{
|
|
|
|
base.Dispose(isDisposing);
|
|
|
|
|
|
|
|
freeModSelectOverlayRegistration?.Dispose();
|
|
|
|
}
|
2021-02-01 13:57:39 +08:00
|
|
|
}
|
|
|
|
}
|