2021-04-08 21:07:00 +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.
|
|
|
|
|
|
|
|
using System;
|
|
|
|
using System.Linq;
|
|
|
|
using osu.Framework.Allocation;
|
2022-09-07 18:12:34 +08:00
|
|
|
using osu.Framework.Audio;
|
2022-08-22 20:52:43 +08:00
|
|
|
using osu.Framework.Extensions.ObjectExtensions;
|
2021-04-08 21:07:00 +08:00
|
|
|
using osu.Framework.Graphics;
|
2021-04-09 17:41:48 +08:00
|
|
|
using osu.Framework.Graphics.Containers;
|
2021-08-25 16:30:13 +08:00
|
|
|
using osu.Game.Graphics;
|
2021-04-26 21:23:44 +08:00
|
|
|
using osu.Game.Online.Multiplayer;
|
2022-02-25 15:03:28 +08:00
|
|
|
using osu.Game.Online.Rooms;
|
2021-04-08 21:07:00 +08:00
|
|
|
using osu.Game.Online.Spectator;
|
2021-04-14 19:39:14 +08:00
|
|
|
using osu.Game.Screens.Play;
|
2021-08-09 18:05:23 +08:00
|
|
|
using osu.Game.Screens.Play.HUD;
|
2021-04-08 21:07:00 +08:00
|
|
|
using osu.Game.Screens.Spectate;
|
2022-02-25 15:03:46 +08:00
|
|
|
using osu.Game.Users;
|
2022-02-25 15:03:28 +08:00
|
|
|
using osuTK;
|
2021-04-08 21:07:00 +08:00
|
|
|
|
|
|
|
namespace osu.Game.Screens.OnlinePlay.Multiplayer.Spectate
|
|
|
|
{
|
2021-04-22 22:52:22 +08:00
|
|
|
/// <summary>
|
|
|
|
/// A <see cref="SpectatorScreen"/> that spectates multiple users in a match.
|
|
|
|
/// </summary>
|
2021-04-22 22:39:02 +08:00
|
|
|
public class MultiSpectatorScreen : SpectatorScreen
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
|
|
|
// Isolates beatmap/ruleset to this screen.
|
|
|
|
public override bool DisallowExternalBeatmapRulesetChanges => true;
|
|
|
|
|
2021-06-08 21:48:07 +08:00
|
|
|
// We are managing our own adjustments. For now, this happens inside the Player instances themselves.
|
2021-09-16 15:08:09 +08:00
|
|
|
public override bool? AllowTrackAdjustments => false;
|
2021-06-08 21:48:07 +08:00
|
|
|
|
2021-04-22 22:52:22 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Whether all spectating players have finished loading.
|
|
|
|
/// </summary>
|
2021-04-08 21:07:00 +08:00
|
|
|
public bool AllPlayersLoaded => instances.All(p => p?.PlayerLoaded == true);
|
|
|
|
|
2022-02-25 15:03:46 +08:00
|
|
|
protected override UserActivity InitialActivity => new UserActivity.SpectatingMultiplayerGame(Beatmap.Value.BeatmapInfo, Ruleset.Value);
|
|
|
|
|
2021-08-25 16:30:13 +08:00
|
|
|
[Resolved]
|
2022-08-22 20:52:43 +08:00
|
|
|
private OsuColour colours { get; set; } = null!;
|
2021-08-25 16:30:13 +08:00
|
|
|
|
2021-04-26 21:23:44 +08:00
|
|
|
[Resolved]
|
2022-08-22 20:52:43 +08:00
|
|
|
private MultiplayerClient multiplayerClient { get; set; } = null!;
|
2021-04-26 21:23:44 +08:00
|
|
|
|
2022-09-07 18:12:34 +08:00
|
|
|
private AudioAdjustments? boundAdjustments;
|
|
|
|
|
2021-04-22 22:52:22 +08:00
|
|
|
private readonly PlayerArea[] instances;
|
2022-08-22 20:52:43 +08:00
|
|
|
private MasterGameplayClockContainer masterClockContainer = null!;
|
2022-08-24 14:07:04 +08:00
|
|
|
private SpectatorSyncManager syncManager = null!;
|
2022-08-22 20:52:43 +08:00
|
|
|
private PlayerGrid grid = null!;
|
|
|
|
private MultiSpectatorLeaderboard leaderboard = null!;
|
|
|
|
private PlayerArea? currentAudioSource;
|
2021-04-08 21:07:00 +08:00
|
|
|
|
2022-02-25 15:03:28 +08:00
|
|
|
private readonly Room room;
|
2021-08-10 17:39:20 +08:00
|
|
|
private readonly MultiplayerRoomUser[] users;
|
|
|
|
|
2021-04-22 22:52:22 +08:00
|
|
|
/// <summary>
|
|
|
|
/// Creates a new <see cref="MultiSpectatorScreen"/>.
|
|
|
|
/// </summary>
|
2022-02-25 15:03:28 +08:00
|
|
|
/// <param name="room">The room.</param>
|
2021-08-10 17:39:20 +08:00
|
|
|
/// <param name="users">The players to spectate.</param>
|
2022-02-25 15:03:28 +08:00
|
|
|
public MultiSpectatorScreen(Room room, MultiplayerRoomUser[] users)
|
2021-08-10 17:39:20 +08:00
|
|
|
: base(users.Select(u => u.UserID).ToArray())
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
2022-02-25 15:03:28 +08:00
|
|
|
this.room = room;
|
2021-08-10 17:39:20 +08:00
|
|
|
this.users = users;
|
|
|
|
|
|
|
|
instances = new PlayerArea[Users.Count];
|
2021-04-08 21:07:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
private void load()
|
|
|
|
{
|
2022-02-25 15:03:28 +08:00
|
|
|
FillFlowContainer leaderboardFlow;
|
2021-08-09 18:05:23 +08:00
|
|
|
Container scoreDisplayContainer;
|
|
|
|
|
2022-08-24 15:08:48 +08:00
|
|
|
InternalChildren = new Drawable[]
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
masterClockContainer = new MasterGameplayClockContainer(Beatmap.Value, 0)
|
2021-04-09 17:41:48 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
Child = new GridContainer
|
2021-04-09 17:41:48 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
RowDimensions = new[] { new Dimension(GridSizeMode.AutoSize) },
|
|
|
|
Content = new[]
|
2021-04-09 17:41:48 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
new Drawable[]
|
2021-04-14 19:39:14 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
scoreDisplayContainer = new Container
|
|
|
|
{
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
AutoSizeAxes = Axes.Y
|
|
|
|
},
|
2021-04-14 19:39:14 +08:00
|
|
|
},
|
2022-08-24 15:08:48 +08:00
|
|
|
new Drawable[]
|
2021-08-09 18:05:23 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
new GridContainer
|
2021-08-09 18:05:23 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
ColumnDimensions = new[] { new Dimension(GridSizeMode.AutoSize) },
|
|
|
|
Content = new[]
|
2021-08-09 18:05:23 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
new Drawable[]
|
2021-08-09 18:05:23 +08:00
|
|
|
{
|
2022-08-24 15:08:48 +08:00
|
|
|
leaderboardFlow = new FillFlowContainer
|
|
|
|
{
|
|
|
|
Anchor = Anchor.CentreLeft,
|
|
|
|
Origin = Anchor.CentreLeft,
|
|
|
|
AutoSizeAxes = Axes.Both,
|
|
|
|
Direction = FillDirection.Vertical,
|
|
|
|
Spacing = new Vector2(5)
|
|
|
|
},
|
|
|
|
grid = new PlayerGrid { RelativeSizeAxes = Axes.Both }
|
|
|
|
}
|
2021-08-09 18:05:23 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-04-14 19:39:14 +08:00
|
|
|
}
|
2021-04-09 17:41:48 +08:00
|
|
|
}
|
2022-08-24 15:08:48 +08:00
|
|
|
},
|
|
|
|
syncManager = new SpectatorSyncManager(masterClockContainer)
|
|
|
|
{
|
|
|
|
ReadyToStart = performInitialSeek,
|
|
|
|
}
|
2021-04-15 18:12:52 +08:00
|
|
|
};
|
|
|
|
|
2021-08-10 17:39:20 +08:00
|
|
|
for (int i = 0; i < Users.Count; i++)
|
2022-08-23 12:52:43 +08:00
|
|
|
grid.Add(instances[i] = new PlayerArea(Users[i], syncManager.CreateManagedClock()));
|
2021-04-16 11:25:29 +08:00
|
|
|
|
2022-05-30 18:18:38 +08:00
|
|
|
LoadComponentAsync(leaderboard = new MultiSpectatorLeaderboard(users)
|
2021-05-03 13:48:04 +08:00
|
|
|
{
|
|
|
|
Expanded = { Value = true },
|
2022-06-24 20:25:23 +08:00
|
|
|
}, _ =>
|
2021-06-29 16:29:25 +08:00
|
|
|
{
|
|
|
|
foreach (var instance in instances)
|
2022-08-24 15:43:26 +08:00
|
|
|
leaderboard.AddClock(instance.UserId, instance.SpectatorPlayerClock);
|
2021-06-29 16:29:25 +08:00
|
|
|
|
2022-02-25 15:03:28 +08:00
|
|
|
leaderboardFlow.Insert(0, leaderboard);
|
2021-08-09 18:05:23 +08:00
|
|
|
|
2021-08-11 13:44:13 +08:00
|
|
|
if (leaderboard.TeamScores.Count == 2)
|
2021-08-09 18:05:23 +08:00
|
|
|
{
|
|
|
|
LoadComponentAsync(new MatchScoreDisplay
|
|
|
|
{
|
|
|
|
Team1Score = { BindTarget = leaderboard.TeamScores.First().Value },
|
|
|
|
Team2Score = { BindTarget = leaderboard.TeamScores.Last().Value },
|
|
|
|
}, scoreDisplayContainer.Add);
|
|
|
|
}
|
2021-06-29 16:29:25 +08:00
|
|
|
});
|
2022-02-25 15:03:28 +08:00
|
|
|
|
|
|
|
LoadComponentAsync(new GameplayChatDisplay(room)
|
|
|
|
{
|
|
|
|
Expanded = { Value = true },
|
|
|
|
}, chat => leaderboardFlow.Insert(1, chat));
|
2021-04-08 21:07:00 +08:00
|
|
|
}
|
|
|
|
|
2021-04-15 18:12:52 +08:00
|
|
|
protected override void LoadComplete()
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
2021-04-15 18:12:52 +08:00
|
|
|
base.LoadComplete();
|
2021-04-14 19:39:14 +08:00
|
|
|
|
2021-04-21 16:13:01 +08:00
|
|
|
masterClockContainer.Reset();
|
2022-09-07 18:12:34 +08:00
|
|
|
|
|
|
|
// Start with adjustments from the first player to keep a sane state.
|
|
|
|
bindAudioAdjustments(instances.First());
|
2021-04-08 21:07:00 +08:00
|
|
|
}
|
|
|
|
|
2021-04-22 21:59:47 +08:00
|
|
|
protected override void Update()
|
|
|
|
{
|
|
|
|
base.Update();
|
|
|
|
|
2022-08-24 15:43:26 +08:00
|
|
|
if (!isCandidateAudioSource(currentAudioSource?.SpectatorPlayerClock))
|
2021-04-22 21:59:47 +08:00
|
|
|
{
|
2022-08-24 15:43:26 +08:00
|
|
|
currentAudioSource = instances.Where(i => isCandidateAudioSource(i.SpectatorPlayerClock))
|
|
|
|
.OrderBy(i => Math.Abs(i.SpectatorPlayerClock.CurrentTime - syncManager.CurrentMasterTime))
|
2021-04-22 21:59:47 +08:00
|
|
|
.FirstOrDefault();
|
|
|
|
|
2022-09-07 18:12:34 +08:00
|
|
|
// Only bind adjustments if there's actually a valid source, else just use the previous ones to ensure no sudden changes to audio.
|
|
|
|
if (currentAudioSource != null)
|
|
|
|
bindAudioAdjustments(currentAudioSource);
|
|
|
|
|
2021-04-22 21:59:47 +08:00
|
|
|
foreach (var instance in instances)
|
2021-04-26 18:01:30 +08:00
|
|
|
instance.Mute = instance != currentAudioSource;
|
2021-04-22 21:59:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-07 18:12:34 +08:00
|
|
|
private void bindAudioAdjustments(PlayerArea first)
|
|
|
|
{
|
|
|
|
if (boundAdjustments != null)
|
|
|
|
masterClockContainer.GameplayAdjustments.UnbindAdjustments(boundAdjustments);
|
|
|
|
|
|
|
|
boundAdjustments = first.SpectatorPlayerClock.GameplayAdjustments;
|
|
|
|
masterClockContainer.GameplayAdjustments.BindAdjustments(boundAdjustments);
|
|
|
|
}
|
|
|
|
|
2022-08-24 14:07:04 +08:00
|
|
|
private bool isCandidateAudioSource(SpectatorPlayerClock? clock)
|
2022-08-24 14:40:47 +08:00
|
|
|
=> clock?.IsRunning == true && !clock.IsCatchingUp && !clock.WaitingOnFrames;
|
2021-04-22 21:59:47 +08:00
|
|
|
|
2022-08-24 14:56:36 +08:00
|
|
|
private void performInitialSeek()
|
2021-06-11 15:25:45 +08:00
|
|
|
{
|
2021-06-11 17:39:50 +08:00
|
|
|
// Seek the master clock to the gameplay time.
|
|
|
|
// This is chosen as the first available frame in the players' replays, which matches the seek by each individual SpectatorPlayer.
|
2021-10-27 12:04:41 +08:00
|
|
|
double startTime = instances.Where(i => i.Score != null)
|
2022-08-22 20:52:43 +08:00
|
|
|
.SelectMany(i => i.Score.AsNonNull().Replay.Frames)
|
2021-10-27 12:04:41 +08:00
|
|
|
.Select(f => f.Time)
|
|
|
|
.DefaultIfEmpty(0)
|
|
|
|
.Min();
|
2021-06-11 15:25:45 +08:00
|
|
|
|
2022-08-24 15:52:36 +08:00
|
|
|
masterClockContainer.Reset(startTime, true);
|
2021-06-11 18:15:53 +08:00
|
|
|
}
|
|
|
|
|
2022-02-09 11:20:07 +08:00
|
|
|
protected override void OnNewPlayingUserState(int userId, SpectatorState spectatorState)
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-10-02 01:08:56 +08:00
|
|
|
protected override void StartGameplay(int userId, SpectatorGameplayState spectatorGameplayState)
|
|
|
|
=> instances.Single(i => i.UserId == userId).LoadScore(spectatorGameplayState.Score);
|
2021-04-08 21:07:00 +08:00
|
|
|
|
2022-08-08 06:37:43 +08:00
|
|
|
protected override void QuitGameplay(int userId)
|
2021-04-08 21:07:00 +08:00
|
|
|
{
|
2021-04-26 20:25:34 +08:00
|
|
|
RemoveUser(userId);
|
2021-08-25 16:30:13 +08:00
|
|
|
|
|
|
|
var instance = instances.Single(i => i.UserId == userId);
|
|
|
|
|
|
|
|
instance.FadeColour(colours.Gray4, 400, Easing.OutQuint);
|
2022-08-24 15:43:26 +08:00
|
|
|
syncManager.RemoveManagedClock(instance.SpectatorPlayerClock);
|
2021-04-08 21:07:00 +08:00
|
|
|
}
|
|
|
|
|
2021-04-26 20:55:38 +08:00
|
|
|
public override bool OnBackButton()
|
|
|
|
{
|
2022-02-08 10:29:39 +08:00
|
|
|
if (multiplayerClient.Room == null)
|
|
|
|
return base.OnBackButton();
|
2021-12-15 16:37:39 +08:00
|
|
|
|
|
|
|
// On a manual exit, set the player back to idle unless gameplay has finished.
|
|
|
|
if (multiplayerClient.Room.State != MultiplayerRoomState.Open)
|
|
|
|
multiplayerClient.ChangeState(MultiplayerUserState.Idle);
|
|
|
|
|
2021-04-26 20:55:38 +08:00
|
|
|
return base.OnBackButton();
|
|
|
|
}
|
2021-04-08 21:07:00 +08:00
|
|
|
}
|
|
|
|
}
|