1
0
mirror of https://github.com/ppy/osu.git synced 2024-11-07 10:17:43 +08:00
osu-lazer/osu.Game/Screens/OnlinePlay/Match/RoomSubScreen.cs

400 lines
15 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.Collections.Generic;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Audio.Sample;
using osu.Framework.Bindables;
2021-08-12 18:51:03 +08:00
using osu.Framework.Extensions.Color4Extensions;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2021-08-12 18:51:03 +08:00
using osu.Framework.Graphics.Shapes;
using osu.Framework.Screens;
using osu.Game.Audio;
using osu.Game.Beatmaps;
2020-12-25 12:38:11 +08:00
using osu.Game.Online.Rooms;
using osu.Game.Overlays;
using osu.Game.Overlays.Mods;
using osu.Game.Rulesets.Mods;
using osu.Game.Screens.OnlinePlay.Match.Components;
namespace osu.Game.Screens.OnlinePlay.Match
{
[Cached(typeof(IPreviewTrackOwner))]
public abstract class RoomSubScreen : OnlinePlaySubScreen, IPreviewTrackOwner
{
[Cached(typeof(IBindable<PlaylistItem>))]
protected readonly Bindable<PlaylistItem> SelectedItem = new Bindable<PlaylistItem>();
public override bool DisallowExternalBeatmapRulesetChanges => true;
/// <summary>
/// A container that provides controls for selection of user mods.
/// This will be shown/hidden automatically when applicable.
/// </summary>
protected Drawable UserModsSection;
2021-01-19 16:11:40 +08:00
private Sample sampleStart;
/// <summary>
/// Any mods applied by/to the local user.
/// </summary>
2021-02-01 16:57:32 +08:00
protected readonly Bindable<IReadOnlyList<Mod>> UserMods = new Bindable<IReadOnlyList<Mod>>(Array.Empty<Mod>());
protected readonly IBindable<long?> RoomId = new Bindable<long?>();
[Resolved]
private MusicController music { get; set; }
[Resolved]
private BeatmapManager beatmapManager { get; set; }
[Resolved(canBeNull: true)]
protected OnlinePlayScreen ParentScreen { get; private set; }
private IBindable<WeakReference<BeatmapSetInfo>> managerUpdated;
[Cached]
2021-08-17 16:05:20 +08:00
protected OnlinePlayBeatmapAvailabilityTracker BeatmapAvailabilityTracker { get; private set; }
2021-04-07 15:45:06 +08:00
protected IBindable<BeatmapAvailability> BeatmapAvailability => BeatmapAvailabilityTracker.Availability;
public readonly Room Room;
2021-08-18 15:35:18 +08:00
private readonly bool allowEdit;
2021-08-17 16:05:20 +08:00
private ModSelectOverlay userModsSelectOverlay;
2021-08-18 14:16:48 +08:00
private RoomSettingsOverlay settingsOverlay;
private Drawable mainContent;
2021-08-17 16:05:20 +08:00
2021-08-18 15:35:18 +08:00
/// <summary>
/// Creates a new <see cref="RoomSubScreen"/>.
/// </summary>
/// <param name="room">The <see cref="Room"/>.</param>
/// <param name="allowEdit">Whether to allow editing room settings post-creation.</param>
protected RoomSubScreen(Room room, bool allowEdit = true)
{
Room = room;
2021-08-18 15:35:18 +08:00
this.allowEdit = allowEdit;
2021-08-17 16:05:20 +08:00
2021-08-12 17:02:00 +08:00
Padding = new MarginPadding { Top = Header.HEIGHT };
2021-08-17 16:05:20 +08:00
BeatmapAvailabilityTracker = new OnlinePlayBeatmapAvailabilityTracker
{
2021-08-17 16:05:20 +08:00
SelectedItem = { BindTarget = SelectedItem }
};
RoomId.BindTo(room.RoomID);
}
[BackgroundDependencyLoader]
private void load(AudioManager audio)
{
sampleStart = audio.Samples.Get(@"SongSelect/confirm-selection");
2021-08-17 16:05:20 +08:00
InternalChildren = new Drawable[]
{
BeatmapAvailabilityTracker,
new GridContainer
{
RelativeSizeAxes = Axes.Both,
2021-08-17 16:13:25 +08:00
RowDimensions = new[]
{
new Dimension(),
new Dimension(GridSizeMode.AutoSize)
},
2021-08-17 16:05:20 +08:00
Content = new[]
{
// Padded main content (drawable room + main content)
new Drawable[]
{
new Container
{
RelativeSizeAxes = Axes.Both,
2021-08-17 16:13:25 +08:00
Padding = new MarginPadding
{
Horizontal = WaveOverlayContainer.WIDTH_PADDING,
Bottom = 30
},
Children = new[]
2021-08-17 16:05:20 +08:00
{
mainContent = new GridContainer
2021-08-17 16:05:20 +08:00
{
RelativeSizeAxes = Axes.Both,
RowDimensions = new[]
2021-08-17 16:05:20 +08:00
{
new Dimension(GridSizeMode.AutoSize),
new Dimension(GridSizeMode.Absolute, 10)
2021-08-17 16:05:20 +08:00
},
Content = new[]
2021-08-17 16:05:20 +08:00
{
new Drawable[]
2021-08-17 16:05:20 +08:00
{
2021-08-18 15:35:18 +08:00
new DrawableMatchRoom(Room, allowEdit)
2021-08-18 14:16:48 +08:00
{
MatchingFilter = true,
OnEdit = () => settingsOverlay.Show()
}
},
null,
new Drawable[]
{
new Container
2021-08-17 16:05:20 +08:00
{
RelativeSizeAxes = Axes.Both,
Children = new[]
2021-08-17 16:05:20 +08:00
{
new Container
2021-08-17 16:05:20 +08:00
{
RelativeSizeAxes = Axes.Both,
Masking = true,
CornerRadius = 10,
Child = new Box
{
RelativeSizeAxes = Axes.Both,
Colour = Color4Extensions.FromHex(@"3e3a44") // This is super temporary.
},
2021-08-17 16:05:20 +08:00
},
2021-08-18 14:54:33 +08:00
new Container
{
RelativeSizeAxes = Axes.Both,
Padding = new MarginPadding(10),
Child = CreateMainContent(),
},
new Container
2021-08-17 16:05:20 +08:00
{
Anchor = Anchor.BottomLeft,
Origin = Anchor.BottomLeft,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Child = userModsSelectOverlay = new UserModSelectOverlay
{
SelectedMods = { BindTarget = UserMods },
IsValidMod = _ => false
}
},
}
2021-08-17 16:05:20 +08:00
}
}
}
},
2021-08-18 14:29:01 +08:00
new Container
{
2021-08-18 14:29:01 +08:00
RelativeSizeAxes = Axes.Both,
// Resolves 1px masking errors between the settings overlay and the room panel.
Padding = new MarginPadding(-1),
Child = settingsOverlay = CreateRoomSettingsOverlay()
2021-08-18 14:29:01 +08:00
}
},
},
2021-08-17 16:05:20 +08:00
},
// Footer
new[]
{
CreateFooter()
}
}
}
};
}
protected override void LoadComplete()
{
base.LoadComplete();
RoomId.BindValueChanged(id =>
{
if (id.NewValue == null)
{
// A new room is being created.
// The main content should be hidden until the settings overlay is hidden, signaling the room is ready to be displayed.
mainContent.Hide();
settingsOverlay.Show();
}
else
{
mainContent.Show();
settingsOverlay.Hide();
}
}, true);
SelectedItem.BindValueChanged(_ => Scheduler.AddOnce(selectedItemChanged));
managerUpdated = beatmapManager.ItemUpdated.GetBoundCopy();
managerUpdated.BindValueChanged(beatmapUpdated);
UserMods.BindValueChanged(_ => Scheduler.AddOnce(UpdateMods));
}
public override bool OnBackButton()
{
if (Room.RoomID.Value == null)
{
// room has not been created yet; exit immediately.
return base.OnBackButton();
}
if (userModsSelectOverlay.State.Value == Visibility.Visible)
{
userModsSelectOverlay.Hide();
return true;
}
2021-08-18 14:16:48 +08:00
if (settingsOverlay.State.Value == Visibility.Visible)
{
2021-08-18 14:16:48 +08:00
settingsOverlay.Hide();
return true;
}
return base.OnBackButton();
}
protected void ShowUserModSelect() => userModsSelectOverlay.Show();
public override void OnEntering(IScreen last)
{
base.OnEntering(last);
beginHandlingTrack();
}
public override void OnSuspending(IScreen next)
{
endHandlingTrack();
base.OnSuspending(next);
}
public override void OnResuming(IScreen last)
{
base.OnResuming(last);
beginHandlingTrack();
Scheduler.AddOnce(UpdateMods);
}
public override bool OnExiting(IScreen next)
{
RoomManager?.PartRoom();
Mods.Value = Array.Empty<Mod>();
endHandlingTrack();
return base.OnExiting(next);
}
2021-04-22 22:37:33 +08:00
protected void StartPlay()
{
sampleStart?.Play();
// fallback is to allow this class to operate when there is no parent OnlineScreen (testing purposes).
var targetScreen = (Screen)ParentScreen ?? this;
2021-05-14 14:32:56 +08:00
targetScreen.Push(CreateGameplayScreen());
}
2021-04-22 22:37:33 +08:00
/// <summary>
/// Creates the gameplay screen to be entered.
/// </summary>
/// <returns>The screen to enter.</returns>
protected abstract Screen CreateGameplayScreen();
private void selectedItemChanged()
{
updateWorkingBeatmap();
2021-02-19 13:36:51 +08:00
var selected = SelectedItem.Value;
if (selected == null)
return;
2021-02-01 17:50:32 +08:00
// Remove any user mods that are no longer allowed.
2021-02-01 16:57:32 +08:00
UserMods.Value = UserMods.Value
2021-02-19 13:36:51 +08:00
.Where(m => selected.AllowedMods.Any(a => m.GetType() == a.GetType()))
2021-02-01 16:57:32 +08:00
.ToList();
UpdateMods();
2021-02-19 13:36:51 +08:00
Ruleset.Value = selected.Ruleset.Value;
if (!selected.AllowedMods.Any())
{
UserModsSection?.Hide();
userModsSelectOverlay.Hide();
userModsSelectOverlay.IsValidMod = _ => false;
}
else
{
UserModsSection?.Show();
2021-02-19 13:36:51 +08:00
userModsSelectOverlay.IsValidMod = m => selected.AllowedMods.Any(a => a.GetType() == m.GetType());
}
}
private void beatmapUpdated(ValueChangedEvent<WeakReference<BeatmapSetInfo>> weakSet) => Schedule(updateWorkingBeatmap);
private void updateWorkingBeatmap()
{
var beatmap = SelectedItem.Value?.Beatmap.Value;
// Retrieve the corresponding local beatmap, since we can't directly use the playlist's beatmap info
var localBeatmap = beatmap == null ? null : beatmapManager.QueryBeatmap(b => b.OnlineBeatmapID == beatmap.OnlineBeatmapID);
Beatmap.Value = beatmapManager.GetWorkingBeatmap(localBeatmap);
}
protected virtual void UpdateMods()
{
if (SelectedItem.Value == null)
return;
2021-02-01 16:57:32 +08:00
Mods.Value = UserMods.Value.Concat(SelectedItem.Value.RequiredMods).ToList();
}
private void beginHandlingTrack()
{
Beatmap.BindValueChanged(applyLoopingToTrack, true);
}
private void endHandlingTrack()
{
Beatmap.ValueChanged -= applyLoopingToTrack;
cancelTrackLooping();
}
private void applyLoopingToTrack(ValueChangedEvent<WorkingBeatmap> _ = null)
{
if (!this.IsCurrentScreen())
return;
var track = Beatmap.Value?.Track;
if (track != null)
{
Beatmap.Value.PrepareTrackForPreviewLooping();
music?.EnsurePlayingSomething();
}
}
private void cancelTrackLooping()
{
var track = Beatmap?.Value?.Track;
if (track != null)
track.Looping = false;
}
2021-08-17 16:05:20 +08:00
protected abstract Drawable CreateMainContent();
protected abstract Drawable CreateFooter();
protected abstract RoomSettingsOverlay CreateRoomSettingsOverlay();
private class UserModSelectOverlay : LocalPlayerModSelectOverlay
{
}
public class UserModSelectButton : PurpleTriangleButton
{
}
}
}