1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-12 08:03:11 +08:00
osu-lazer/osu.Game/Screens/Spectate/SpectatorScreen.cs

288 lines
9.6 KiB
C#
Raw Normal View History

2021-04-01 21:02:32 +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;
2021-04-01 21:02:32 +08:00
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Extensions;
using osu.Framework.Extensions.ObjectExtensions;
2021-04-01 21:02:32 +08:00
using osu.Game.Beatmaps;
using osu.Game.Database;
using osu.Game.Online.API.Requests.Responses;
2021-04-01 21:02:32 +08:00
using osu.Game.Online.Spectator;
using osu.Game.Replays;
using osu.Game.Rulesets;
using osu.Game.Scoring;
using Realms;
2021-04-01 21:02:32 +08:00
namespace osu.Game.Screens.Spectate
{
/// <summary>
/// A <see cref="OsuScreen"/> which spectates one or more users.
/// </summary>
2022-11-24 13:32:20 +08:00
public abstract partial class SpectatorScreen : OsuScreen
2021-04-01 21:02:32 +08:00
{
protected IReadOnlyList<int> Users => users;
private readonly List<int> users = new List<int>();
2021-04-01 21:02:32 +08:00
[Resolved]
2023-11-22 14:14:25 +08:00
private BeatmapManager beatmaps { get; set; } = null!;
[Resolved]
private RulesetStore rulesets { get; set; } = null!;
2021-04-01 21:02:32 +08:00
[Resolved]
2023-11-22 14:14:25 +08:00
private SpectatorClient spectatorClient { get; set; } = null!;
2021-04-01 21:02:32 +08:00
[Resolved]
2023-11-22 14:14:25 +08:00
private UserLookupCache userLookupCache { get; set; } = null!;
2021-04-01 21:02:32 +08:00
[Resolved]
2023-11-22 14:14:25 +08:00
private RealmAccess realm { get; set; } = null!;
2021-04-01 21:02:32 +08:00
private readonly IBindableDictionary<int, SpectatorState> userStates = new BindableDictionary<int, SpectatorState>();
private readonly Dictionary<int, APIUser> userMap = new Dictionary<int, APIUser>();
private readonly Dictionary<int, SpectatorGameplayState> gameplayStates = new Dictionary<int, SpectatorGameplayState>();
2021-04-01 21:02:32 +08:00
2023-11-22 14:14:25 +08:00
private IDisposable? realmSubscription;
2021-04-01 21:02:32 +08:00
/// <summary>
/// Creates a new <see cref="SpectatorScreen"/>.
/// </summary>
/// <param name="users">The users to spectate.</param>
protected SpectatorScreen(params int[] users)
2021-04-01 21:02:32 +08:00
{
this.users.AddRange(users);
2021-04-01 21:02:32 +08:00
}
protected override void LoadComplete()
{
base.LoadComplete();
userLookupCache.GetUsersAsync(users.ToArray()).ContinueWith(task => Schedule(() =>
{
2022-01-06 21:54:43 +08:00
var foundUsers = task.GetResultSafely();
foreach (var u in foundUsers)
{
if (u == null)
continue;
2021-05-20 17:52:20 +08:00
userMap[u.Id] = u;
}
2021-05-20 17:52:20 +08:00
2022-02-09 11:09:04 +08:00
userStates.BindTo(spectatorClient.WatchedUserStates);
userStates.BindCollectionChanged(onUserStatesChanged, true);
realmSubscription = realm.RegisterForNotifications(
realm => realm.All<BeatmapSetInfo>().Where(s => !s.DeletePending), beatmapsChanged);
foreach ((int id, var _) in userMap)
2021-05-20 17:52:20 +08:00
spectatorClient.WatchUser(id);
}));
}
2021-04-01 21:02:32 +08:00
2023-11-22 14:14:25 +08:00
private void beatmapsChanged(IRealmCollection<BeatmapSetInfo> items, ChangeSet? changes)
{
if (changes?.InsertedIndices == null) return;
foreach (int c in changes.InsertedIndices) beatmapUpdated(items[c]);
}
private void beatmapUpdated(BeatmapSetInfo beatmapSet)
2021-04-01 21:02:32 +08:00
{
foreach ((int userId, _) in userMap)
2021-04-01 21:02:32 +08:00
{
if (!userStates.TryGetValue(userId, out var userState))
2021-05-20 17:52:20 +08:00
continue;
if (beatmapSet.Beatmaps.Any(b => b.OnlineID == userState.BeatmapID))
startGameplay(userId);
2021-04-01 21:02:32 +08:00
}
}
2023-11-22 14:14:25 +08:00
private void onUserStatesChanged(object? sender, NotifyDictionaryChangedEventArgs<int, SpectatorState> e)
{
switch (e.Action)
{
case NotifyDictionaryChangedAction.Add:
2022-02-03 11:44:33 +08:00
case NotifyDictionaryChangedAction.Replace:
foreach ((int userId, SpectatorState state) in e.NewItems.AsNonNull())
onUserStateChanged(userId, state);
break;
}
}
private void onUserStateChanged(int userId, SpectatorState newState)
2021-04-01 21:02:32 +08:00
{
if (newState.RulesetID == null || newState.BeatmapID == null)
2021-04-01 21:02:32 +08:00
return;
2021-05-20 17:52:20 +08:00
if (!userMap.ContainsKey(userId))
return;
2021-04-01 21:02:32 +08:00
2022-02-03 11:44:33 +08:00
switch (newState.State)
{
2022-02-09 11:09:04 +08:00
case SpectatedUserState.Playing:
OnNewPlayingUserState(userId, newState);
startGameplay(userId);
2022-02-03 11:44:33 +08:00
break;
case SpectatedUserState.Passed:
markReceivedAllFrames(userId);
PassGameplay(userId);
break;
case SpectatedUserState.Failed:
failGameplay(userId);
break;
case SpectatedUserState.Quit:
quitGameplay(userId);
break;
}
2021-04-01 21:02:32 +08:00
}
private void startGameplay(int userId)
2021-04-01 21:02:32 +08:00
{
2021-05-20 17:52:20 +08:00
Debug.Assert(userMap.ContainsKey(userId));
2021-04-01 21:02:32 +08:00
2021-05-20 17:52:20 +08:00
var user = userMap[userId];
var spectatorState = userStates[userId];
2021-04-01 21:02:32 +08:00
var resolvedRuleset = rulesets.AvailableRulesets.FirstOrDefault(r => r.OnlineID == spectatorState.RulesetID)?.CreateInstance();
2021-05-20 17:52:20 +08:00
if (resolvedRuleset == null)
return;
2021-04-01 21:02:32 +08:00
var resolvedBeatmap = beatmaps.QueryBeatmap(b => b.OnlineID == spectatorState.BeatmapID);
2021-05-20 17:52:20 +08:00
if (resolvedBeatmap == null)
return;
2021-04-01 21:02:32 +08:00
2021-05-20 17:52:20 +08:00
var score = new Score
{
ScoreInfo = new ScoreInfo
2021-04-01 21:02:32 +08:00
{
BeatmapInfo = resolvedBeatmap,
2021-05-20 17:52:20 +08:00
User = user,
Mods = spectatorState.Mods.Select(m => m.ToMod(resolvedRuleset)).ToArray(),
Ruleset = resolvedRuleset.RulesetInfo,
},
Replay = new Replay { HasReceivedAllFrames = false },
};
var gameplayState = new SpectatorGameplayState(score, resolvedRuleset, beatmaps.GetWorkingBeatmap(resolvedBeatmap));
2021-05-20 17:52:20 +08:00
gameplayStates[userId] = gameplayState;
StartGameplay(userId, gameplayState);
2021-04-01 21:02:32 +08:00
}
/// <summary>
/// Marks an existing gameplay session as received all frames.
/// </summary>
private void markReceivedAllFrames(int userId)
{
if (gameplayStates.TryGetValue(userId, out var gameplayState))
gameplayState.Score.Replay.HasReceivedAllFrames = true;
}
private void failGameplay(int userId)
{
if (!userMap.ContainsKey(userId))
return;
if (!gameplayStates.ContainsKey(userId))
return;
markReceivedAllFrames(userId);
gameplayStates.Remove(userId);
FailGameplay(userId);
}
private void quitGameplay(int userId)
{
if (!userMap.ContainsKey(userId))
return;
if (!gameplayStates.ContainsKey(userId))
return;
markReceivedAllFrames(userId);
gameplayStates.Remove(userId);
QuitGameplay(userId);
}
2021-04-01 21:02:32 +08:00
/// <summary>
/// Invoked when a spectated user's state has changed to a new state indicating the player is currently playing.
/// Thread safety is not guaranteed should be scheduled as required.
2021-04-01 21:02:32 +08:00
/// </summary>
/// <param name="userId">The user whose state has changed.</param>
/// <param name="spectatorState">The new state.</param>
2023-11-22 14:14:25 +08:00
protected abstract void OnNewPlayingUserState(int userId, SpectatorState spectatorState);
2021-04-01 21:02:32 +08:00
/// <summary>
/// Starts gameplay for a user.
/// Thread safety is not guaranteed should be scheduled as required.
2021-04-01 21:02:32 +08:00
/// </summary>
/// <param name="userId">The user to start gameplay for.</param>
/// <param name="spectatorGameplayState">The gameplay state.</param>
2023-11-22 14:14:25 +08:00
protected abstract void StartGameplay(int userId, SpectatorGameplayState spectatorGameplayState);
2021-04-01 21:02:32 +08:00
/// <summary>
/// Fired when a user passes gameplay.
/// </summary>
/// <param name="userId">The user which passed.</param>
protected virtual void PassGameplay(int userId) { }
2021-04-01 21:02:32 +08:00
/// <summary>
/// Quits gameplay for a user.
/// Thread safety is not guaranteed should be scheduled as required.
2021-04-01 21:02:32 +08:00
/// </summary>
/// <param name="userId">The user to quit gameplay for.</param>
protected abstract void QuitGameplay(int userId);
2021-04-01 21:02:32 +08:00
/// <summary>
/// Fails gameplay for a user.
/// Thread safety is not guaranteed should be scheduled as required.
/// </summary>
/// <param name="userId">The user to fail gameplay for.</param>
protected abstract void FailGameplay(int userId);
/// <summary>
/// Stops spectating a user.
/// </summary>
/// <param name="userId">The user to stop spectating.</param>
protected void RemoveUser(int userId)
{
if (!userStates.ContainsKey(userId))
return;
quitGameplay(userId);
users.Remove(userId);
2021-05-20 17:52:20 +08:00
userMap.Remove(userId);
2021-05-20 17:52:20 +08:00
spectatorClient.StopWatchingUser(userId);
}
2021-04-01 21:02:32 +08:00
protected override void Dispose(bool isDisposing)
{
base.Dispose(isDisposing);
2023-11-22 14:14:25 +08:00
if (spectatorClient.IsNotNull())
2021-04-01 21:02:32 +08:00
{
foreach ((int userId, var _) in userMap)
2021-05-20 17:52:20 +08:00
spectatorClient.StopWatchingUser(userId);
2021-04-01 21:02:32 +08:00
}
realmSubscription?.Dispose();
2021-04-01 21:02:32 +08:00
}
}
}