mirror of
https://github.com/ppy/osu.git
synced 2024-12-14 11:35:35 +08:00
Combine test spectator streaming client implementations
This commit is contained in:
parent
9b09361cc9
commit
21fc0ba43b
@ -1,35 +1,32 @@
|
|||||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
// 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.
|
// See the LICENCE file in the repository root for full licence text.
|
||||||
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using NUnit.Framework;
|
using NUnit.Framework;
|
||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
using osu.Framework.Bindables;
|
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Screens;
|
using osu.Framework.Screens;
|
||||||
using osu.Framework.Testing;
|
using osu.Framework.Testing;
|
||||||
using osu.Framework.Utils;
|
|
||||||
using osu.Game.Beatmaps;
|
using osu.Game.Beatmaps;
|
||||||
using osu.Game.Database;
|
using osu.Game.Database;
|
||||||
using osu.Game.Online;
|
|
||||||
using osu.Game.Online.Spectator;
|
using osu.Game.Online.Spectator;
|
||||||
using osu.Game.Replays.Legacy;
|
|
||||||
using osu.Game.Rulesets.Osu;
|
using osu.Game.Rulesets.Osu;
|
||||||
using osu.Game.Rulesets.Osu.Replays;
|
using osu.Game.Rulesets.Osu.Replays;
|
||||||
using osu.Game.Rulesets.UI;
|
using osu.Game.Rulesets.UI;
|
||||||
using osu.Game.Scoring;
|
|
||||||
using osu.Game.Screens.Play;
|
using osu.Game.Screens.Play;
|
||||||
using osu.Game.Tests.Beatmaps.IO;
|
using osu.Game.Tests.Beatmaps.IO;
|
||||||
using osu.Game.Tests.Visual.Multiplayer;
|
using osu.Game.Tests.Visual.Multiplayer;
|
||||||
|
using osu.Game.Tests.Visual.Spectator;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Gameplay
|
namespace osu.Game.Tests.Visual.Gameplay
|
||||||
{
|
{
|
||||||
public class TestSceneSpectator : ScreenTestScene
|
public class TestSceneSpectator : ScreenTestScene
|
||||||
{
|
{
|
||||||
|
private readonly User streamingUser = new User { Id = MultiplayerTestScene.PLAYER_1_ID, Username = "Test user" };
|
||||||
|
|
||||||
[Cached(typeof(SpectatorStreamingClient))]
|
[Cached(typeof(SpectatorStreamingClient))]
|
||||||
private TestSpectatorStreamingClient testSpectatorStreamingClient = new TestSpectatorStreamingClient();
|
private TestSpectatorStreamingClient testSpectatorStreamingClient = new TestSpectatorStreamingClient();
|
||||||
|
|
||||||
@ -215,9 +212,9 @@ namespace osu.Game.Tests.Visual.Gameplay
|
|||||||
|
|
||||||
private void waitForPlayer() => AddUntilStep("wait for player", () => Stack.CurrentScreen is Player);
|
private void waitForPlayer() => AddUntilStep("wait for player", () => Stack.CurrentScreen is Player);
|
||||||
|
|
||||||
private void start(int? beatmapId = null) => AddStep("start play", () => testSpectatorStreamingClient.StartPlay(beatmapId ?? importedBeatmapId));
|
private void start(int? beatmapId = null) => AddStep("start play", () => testSpectatorStreamingClient.StartPlay(streamingUser.Id, beatmapId ?? importedBeatmapId));
|
||||||
|
|
||||||
private void finish(int? beatmapId = null) => AddStep("end play", () => testSpectatorStreamingClient.EndPlay(beatmapId ?? importedBeatmapId));
|
private void finish(int? beatmapId = null) => AddStep("end play", () => testSpectatorStreamingClient.EndPlay(streamingUser.Id, beatmapId ?? importedBeatmapId));
|
||||||
|
|
||||||
private void checkPaused(bool state) =>
|
private void checkPaused(bool state) =>
|
||||||
AddUntilStep($"game is {(state ? "paused" : "playing")}", () => player.ChildrenOfType<DrawableRuleset>().First().IsPaused.Value == state);
|
AddUntilStep($"game is {(state ? "paused" : "playing")}", () => player.ChildrenOfType<DrawableRuleset>().First().IsPaused.Value == state);
|
||||||
@ -226,89 +223,17 @@ namespace osu.Game.Tests.Visual.Gameplay
|
|||||||
{
|
{
|
||||||
AddStep("send frames", () =>
|
AddStep("send frames", () =>
|
||||||
{
|
{
|
||||||
testSpectatorStreamingClient.SendFrames(nextFrame, count);
|
testSpectatorStreamingClient.SendFrames(streamingUser.Id, nextFrame, count);
|
||||||
nextFrame += count;
|
nextFrame += count;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void loadSpectatingScreen()
|
private void loadSpectatingScreen()
|
||||||
{
|
{
|
||||||
AddStep("load screen", () => LoadScreen(spectatorScreen = new SoloSpectator(testSpectatorStreamingClient.StreamingUser)));
|
AddStep("load screen", () => LoadScreen(spectatorScreen = new SoloSpectator(streamingUser)));
|
||||||
AddUntilStep("wait for screen load", () => spectatorScreen.LoadState == LoadState.Loaded);
|
AddUntilStep("wait for screen load", () => spectatorScreen.LoadState == LoadState.Loaded);
|
||||||
}
|
}
|
||||||
|
|
||||||
public class TestSpectatorStreamingClient : SpectatorStreamingClient
|
|
||||||
{
|
|
||||||
public readonly User StreamingUser = new User { Id = MultiplayerTestScene.PLAYER_1_ID, Username = "Test user" };
|
|
||||||
|
|
||||||
public new BindableList<int> PlayingUsers => (BindableList<int>)base.PlayingUsers;
|
|
||||||
|
|
||||||
private int beatmapId;
|
|
||||||
|
|
||||||
public TestSpectatorStreamingClient()
|
|
||||||
: base(new DevelopmentEndpointConfiguration())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public void StartPlay(int beatmapId)
|
|
||||||
{
|
|
||||||
this.beatmapId = beatmapId;
|
|
||||||
sendState(beatmapId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void EndPlay(int beatmapId)
|
|
||||||
{
|
|
||||||
((ISpectatorClient)this).UserFinishedPlaying(StreamingUser.Id, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
sentState = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private bool sentState;
|
|
||||||
|
|
||||||
public void SendFrames(int index, int count)
|
|
||||||
{
|
|
||||||
var frames = new List<LegacyReplayFrame>();
|
|
||||||
|
|
||||||
for (int i = index; i < index + count; i++)
|
|
||||||
{
|
|
||||||
var buttonState = i == index + count - 1 ? ReplayButtonState.None : ReplayButtonState.Left1;
|
|
||||||
|
|
||||||
frames.Add(new LegacyReplayFrame(i * 100, RNG.Next(0, 512), RNG.Next(0, 512), buttonState));
|
|
||||||
}
|
|
||||||
|
|
||||||
var bundle = new FrameDataBundle(new ScoreInfo(), frames);
|
|
||||||
((ISpectatorClient)this).UserSentFrames(StreamingUser.Id, bundle);
|
|
||||||
|
|
||||||
if (!sentState)
|
|
||||||
sendState(beatmapId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void WatchUser(int userId)
|
|
||||||
{
|
|
||||||
if (!PlayingUsers.Contains(userId) && sentState)
|
|
||||||
{
|
|
||||||
// usually the server would do this.
|
|
||||||
sendState(beatmapId);
|
|
||||||
}
|
|
||||||
|
|
||||||
base.WatchUser(userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void sendState(int beatmapId)
|
|
||||||
{
|
|
||||||
sentState = true;
|
|
||||||
((ISpectatorClient)this).UserBeganPlaying(StreamingUser.Id, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal class TestUserLookupCache : UserLookupCache
|
internal class TestUserLookupCache : UserLookupCache
|
||||||
{
|
{
|
||||||
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default) => Task.FromResult(new User
|
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default) => Task.FromResult(new User
|
||||||
|
@ -11,15 +11,12 @@ using osu.Framework.Graphics;
|
|||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Testing;
|
using osu.Framework.Testing;
|
||||||
using osu.Framework.Timing;
|
using osu.Framework.Timing;
|
||||||
using osu.Framework.Utils;
|
|
||||||
using osu.Game.Database;
|
using osu.Game.Database;
|
||||||
using osu.Game.Online;
|
|
||||||
using osu.Game.Online.Spectator;
|
using osu.Game.Online.Spectator;
|
||||||
using osu.Game.Replays.Legacy;
|
|
||||||
using osu.Game.Rulesets.Osu.Scoring;
|
using osu.Game.Rulesets.Osu.Scoring;
|
||||||
using osu.Game.Scoring;
|
|
||||||
using osu.Game.Screens.OnlinePlay.Multiplayer.Spectate;
|
using osu.Game.Screens.OnlinePlay.Multiplayer.Spectate;
|
||||||
using osu.Game.Screens.Play.HUD;
|
using osu.Game.Screens.Play.HUD;
|
||||||
|
using osu.Game.Tests.Visual.Spectator;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Multiplayer
|
namespace osu.Game.Tests.Visual.Multiplayer
|
||||||
@ -149,71 +146,6 @@ namespace osu.Game.Tests.Visual.Multiplayer
|
|||||||
private void assertCombo(int userId, int expectedCombo)
|
private void assertCombo(int userId, int expectedCombo)
|
||||||
=> AddUntilStep($"player {userId} has {expectedCombo} combo", () => this.ChildrenOfType<GameplayLeaderboardScore>().Single(s => s.User?.Id == userId).Combo.Value == expectedCombo);
|
=> AddUntilStep($"player {userId} has {expectedCombo} combo", () => this.ChildrenOfType<GameplayLeaderboardScore>().Single(s => s.User?.Id == userId).Combo.Value == expectedCombo);
|
||||||
|
|
||||||
private class TestSpectatorStreamingClient : SpectatorStreamingClient
|
|
||||||
{
|
|
||||||
private readonly Dictionary<int, int> userBeatmapDictionary = new Dictionary<int, int>();
|
|
||||||
private readonly Dictionary<int, bool> userSentStateDictionary = new Dictionary<int, bool>();
|
|
||||||
|
|
||||||
public TestSpectatorStreamingClient()
|
|
||||||
: base(new DevelopmentEndpointConfiguration())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public void StartPlay(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
userBeatmapDictionary[userId] = beatmapId;
|
|
||||||
userSentStateDictionary[userId] = false;
|
|
||||||
sendState(userId, beatmapId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void EndPlay(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
((ISpectatorClient)this).UserFinishedPlaying(userId, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
userSentStateDictionary[userId] = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendFrames(int userId, int index, int count)
|
|
||||||
{
|
|
||||||
var frames = new List<LegacyReplayFrame>();
|
|
||||||
|
|
||||||
for (int i = index; i < index + count; i++)
|
|
||||||
{
|
|
||||||
var buttonState = i == index + count - 1 ? ReplayButtonState.None : ReplayButtonState.Left1;
|
|
||||||
frames.Add(new LegacyReplayFrame(i * 100, RNG.Next(0, 512), RNG.Next(0, 512), buttonState));
|
|
||||||
}
|
|
||||||
|
|
||||||
var bundle = new FrameDataBundle(new ScoreInfo { Combo = index + count }, frames);
|
|
||||||
((ISpectatorClient)this).UserSentFrames(userId, bundle);
|
|
||||||
if (!userSentStateDictionary[userId])
|
|
||||||
sendState(userId, userBeatmapDictionary[userId]);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void WatchUser(int userId)
|
|
||||||
{
|
|
||||||
if (userSentStateDictionary[userId])
|
|
||||||
{
|
|
||||||
// usually the server would do this.
|
|
||||||
sendState(userId, userBeatmapDictionary[userId]);
|
|
||||||
}
|
|
||||||
|
|
||||||
base.WatchUser(userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void sendState(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
((ISpectatorClient)this).UserBeganPlaying(userId, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
userSentStateDictionary[userId] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class TestUserLookupCache : UserLookupCache
|
private class TestUserLookupCache : UserLookupCache
|
||||||
{
|
{
|
||||||
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default)
|
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default)
|
||||||
|
@ -9,16 +9,13 @@ using NUnit.Framework;
|
|||||||
using osu.Framework.Allocation;
|
using osu.Framework.Allocation;
|
||||||
using osu.Framework.Graphics;
|
using osu.Framework.Graphics;
|
||||||
using osu.Framework.Testing;
|
using osu.Framework.Testing;
|
||||||
using osu.Framework.Utils;
|
|
||||||
using osu.Game.Beatmaps;
|
using osu.Game.Beatmaps;
|
||||||
using osu.Game.Database;
|
using osu.Game.Database;
|
||||||
using osu.Game.Online;
|
|
||||||
using osu.Game.Online.Spectator;
|
using osu.Game.Online.Spectator;
|
||||||
using osu.Game.Replays.Legacy;
|
|
||||||
using osu.Game.Scoring;
|
|
||||||
using osu.Game.Screens.OnlinePlay.Multiplayer.Spectate;
|
using osu.Game.Screens.OnlinePlay.Multiplayer.Spectate;
|
||||||
using osu.Game.Screens.Play;
|
using osu.Game.Screens.Play;
|
||||||
using osu.Game.Tests.Beatmaps.IO;
|
using osu.Game.Tests.Beatmaps.IO;
|
||||||
|
using osu.Game.Tests.Visual.Spectator;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Multiplayer
|
namespace osu.Game.Tests.Visual.Multiplayer
|
||||||
@ -301,76 +298,6 @@ namespace osu.Game.Tests.Visual.Multiplayer
|
|||||||
|
|
||||||
private PlayerArea getInstance(int userId) => spectatorScreen.ChildrenOfType<PlayerArea>().Single(p => p.UserId == userId);
|
private PlayerArea getInstance(int userId) => spectatorScreen.ChildrenOfType<PlayerArea>().Single(p => p.UserId == userId);
|
||||||
|
|
||||||
public class TestSpectatorStreamingClient : SpectatorStreamingClient
|
|
||||||
{
|
|
||||||
private readonly Dictionary<int, int> userBeatmapDictionary = new Dictionary<int, int>();
|
|
||||||
private readonly Dictionary<int, bool> userSentStateDictionary = new Dictionary<int, bool>();
|
|
||||||
|
|
||||||
public TestSpectatorStreamingClient()
|
|
||||||
: base(new DevelopmentEndpointConfiguration())
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public void StartPlay(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
userBeatmapDictionary[userId] = beatmapId;
|
|
||||||
userSentStateDictionary[userId] = false;
|
|
||||||
|
|
||||||
sendState(userId, beatmapId);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void EndPlay(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
((ISpectatorClient)this).UserFinishedPlaying(userId, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
userSentStateDictionary[userId] = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendFrames(int userId, int index, int count)
|
|
||||||
{
|
|
||||||
var frames = new List<LegacyReplayFrame>();
|
|
||||||
|
|
||||||
for (int i = index; i < index + count; i++)
|
|
||||||
{
|
|
||||||
var buttonState = i == index + count - 1 ? ReplayButtonState.None : ReplayButtonState.Left1;
|
|
||||||
|
|
||||||
frames.Add(new LegacyReplayFrame(i * 100, RNG.Next(0, 512), RNG.Next(0, 512), buttonState));
|
|
||||||
}
|
|
||||||
|
|
||||||
var bundle = new FrameDataBundle(new ScoreInfo { Combo = index + count }, frames);
|
|
||||||
((ISpectatorClient)this).UserSentFrames(userId, bundle);
|
|
||||||
|
|
||||||
if (!userSentStateDictionary[userId])
|
|
||||||
sendState(userId, userBeatmapDictionary[userId]);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void WatchUser(int userId)
|
|
||||||
{
|
|
||||||
if (!PlayingUsers.Contains(userId) && userSentStateDictionary.TryGetValue(userId, out var sent) && sent)
|
|
||||||
{
|
|
||||||
// usually the server would do this.
|
|
||||||
sendState(userId, userBeatmapDictionary[userId]);
|
|
||||||
}
|
|
||||||
|
|
||||||
base.WatchUser(userId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void sendState(int userId, int beatmapId)
|
|
||||||
{
|
|
||||||
((ISpectatorClient)this).UserBeganPlaying(userId, new SpectatorState
|
|
||||||
{
|
|
||||||
BeatmapID = beatmapId,
|
|
||||||
RulesetID = 0,
|
|
||||||
});
|
|
||||||
|
|
||||||
userSentStateDictionary[userId] = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
internal class TestUserLookupCache : UserLookupCache
|
internal class TestUserLookupCache : UserLookupCache
|
||||||
{
|
{
|
||||||
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default)
|
protected override Task<User> ComputeValueAsync(int lookup, CancellationToken token = default)
|
||||||
|
@ -12,7 +12,7 @@ using osu.Framework.Testing;
|
|||||||
using osu.Game.Database;
|
using osu.Game.Database;
|
||||||
using osu.Game.Online.Spectator;
|
using osu.Game.Online.Spectator;
|
||||||
using osu.Game.Overlays.Dashboard;
|
using osu.Game.Overlays.Dashboard;
|
||||||
using osu.Game.Tests.Visual.Gameplay;
|
using osu.Game.Tests.Visual.Spectator;
|
||||||
using osu.Game.Users;
|
using osu.Game.Users;
|
||||||
|
|
||||||
namespace osu.Game.Tests.Visual.Online
|
namespace osu.Game.Tests.Visual.Online
|
||||||
@ -20,7 +20,7 @@ namespace osu.Game.Tests.Visual.Online
|
|||||||
public class TestSceneCurrentlyPlayingDisplay : OsuTestScene
|
public class TestSceneCurrentlyPlayingDisplay : OsuTestScene
|
||||||
{
|
{
|
||||||
[Cached(typeof(SpectatorStreamingClient))]
|
[Cached(typeof(SpectatorStreamingClient))]
|
||||||
private TestSceneSpectator.TestSpectatorStreamingClient testSpectatorStreamingClient = new TestSceneSpectator.TestSpectatorStreamingClient();
|
private TestSpectatorStreamingClient testSpectatorStreamingClient = new TestSpectatorStreamingClient();
|
||||||
|
|
||||||
private CurrentlyPlayingDisplay currentlyPlaying;
|
private CurrentlyPlayingDisplay currentlyPlaying;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user