1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-06 00:52:54 +08:00
osu-lazer/osu.Game.Tests/Visual/Playlists/TestScenePlaylistsResultsScreen.cs

431 lines
17 KiB
C#
Raw Normal View History

2020-05-28 19:08:45 +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
2020-05-28 19:08:45 +08:00
using System;
using System.Collections.Generic;
2020-07-31 20:39:50 +08:00
using System.Linq;
using System.Net;
using JetBrains.Annotations;
using Newtonsoft.Json.Linq;
2020-05-28 19:08:45 +08:00
using NUnit.Framework;
2020-07-31 20:39:50 +08:00
using osu.Framework.Graphics.Containers;
using osu.Framework.Testing;
using osu.Game.Graphics.Containers;
using osu.Game.Graphics.UserInterface;
2020-05-28 19:08:45 +08:00
using osu.Game.Online.API;
2020-07-31 20:39:50 +08:00
using osu.Game.Online.API.Requests;
using osu.Game.Online.API.Requests.Responses;
using osu.Game.Online.Placeholders;
2020-12-25 12:38:11 +08:00
using osu.Game.Online.Rooms;
2020-05-28 19:08:45 +08:00
using osu.Game.Rulesets.Osu;
using osu.Game.Rulesets.Scoring;
using osu.Game.Scoring;
using osu.Game.Screens.OnlinePlay.Playlists;
2020-07-31 20:39:50 +08:00
using osu.Game.Screens.Ranking;
2020-05-28 19:08:45 +08:00
using osu.Game.Tests.Beatmaps;
using osu.Game.Tests.Resources;
2020-05-28 19:08:45 +08:00
namespace osu.Game.Tests.Visual.Playlists
2020-05-28 19:08:45 +08:00
{
2022-11-24 13:32:20 +08:00
public partial class TestScenePlaylistsResultsScreen : ScreenTestScene
2020-05-28 19:08:45 +08:00
{
2020-07-31 20:39:50 +08:00
private const int scores_per_result = 10;
private const int real_user_position = 200;
2020-07-31 20:39:50 +08:00
private TestResultsScreen resultsScreen;
private int lowestScoreId; // Score ID of the lowest score in the list.
private int highestScoreId; // Score ID of the highest score in the list.
2020-07-31 20:39:50 +08:00
private bool requestComplete;
private int totalCount;
private ScoreInfo userScore;
2020-06-09 17:53:55 +08:00
2024-01-09 20:34:40 +08:00
[SetUpSteps]
public override void SetUpSteps()
{
base.SetUpSteps();
// Previous test instances of the results screen may still exist at this point so wait for
// those screens to be cleaned up by the base SetUpSteps before re-initialising test state.
// The screen also holds a leased Beatmap bindable so reassigning it must happen after
// the screen has been exited.
AddStep("initialise user scores and beatmap", () =>
{
lowestScoreId = 1;
highestScoreId = 1;
requestComplete = false;
totalCount = 0;
userScore = TestResources.CreateTestScoreInfo();
userScore.TotalScore = 0;
userScore.Statistics = new Dictionary<HitResult, int>();
userScore.MaximumStatistics = new Dictionary<HitResult, int>();
// Beatmap is required to be an actual beatmap so the scores can get their scores correctly
// calculated for standardised scoring, else the tests that rely on ordering will fall over.
Beatmap.Value = CreateWorkingBeatmap(Ruleset.Value);
});
}
2020-05-28 19:08:45 +08:00
[Test]
2020-07-31 20:39:50 +08:00
public void TestShowWithUserScore()
2020-05-28 21:25:00 +08:00
{
AddStep("bind user score info handler", () => bindHandler(userScore: userScore));
2020-07-31 20:39:50 +08:00
createResults(() => userScore);
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
2021-12-10 14:37:12 +08:00
AddAssert("user score selected", () => this.ChildrenOfType<ScorePanel>().Single(p => p.Score.OnlineID == userScore.OnlineID).State == PanelState.Expanded);
AddAssert($"score panel position is {real_user_position}",
() => this.ChildrenOfType<ScorePanel>().Single(p => p.Score.OnlineID == userScore.OnlineID).ScorePosition.Value == real_user_position);
2020-05-28 21:25:00 +08:00
}
[Test]
2020-07-31 20:39:50 +08:00
public void TestShowNullUserScore()
2020-05-28 21:25:00 +08:00
{
AddStep("bind user score info handler", () => bindHandler());
2020-07-31 20:39:50 +08:00
createResults();
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
AddAssert("top score selected", () => this.ChildrenOfType<ScorePanel>().OrderByDescending(p => p.Score.TotalScore).First().State == PanelState.Expanded);
2020-06-09 17:53:55 +08:00
}
[Test]
2020-07-31 20:39:50 +08:00
public void TestShowUserScoreWithDelay()
{
AddStep("bind user score info handler", () => bindHandler(true, userScore));
2020-07-31 20:39:50 +08:00
createResults(() => userScore);
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
AddAssert("more than 1 panel displayed", () => this.ChildrenOfType<ScorePanel>().Count() > 1);
2021-12-10 14:37:12 +08:00
AddAssert("user score selected", () => this.ChildrenOfType<ScorePanel>().Single(p => p.Score.OnlineID == userScore.OnlineID).State == PanelState.Expanded);
2020-07-31 20:39:50 +08:00
}
[Test]
public void TestShowNullUserScoreWithDelay()
2020-06-09 17:53:55 +08:00
{
AddStep("bind delayed handler", () => bindHandler(true));
2020-07-31 20:39:50 +08:00
createResults();
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
AddAssert("top score selected", () => this.ChildrenOfType<ScorePanel>().OrderByDescending(p => p.Score.TotalScore).First().State == PanelState.Expanded);
2020-05-28 21:25:00 +08:00
}
2020-07-31 20:39:50 +08:00
[Test]
public void TestFetchWhenScrolledToTheRight()
{
AddStep("bind delayed handler", () => bindHandler(true));
2020-07-31 20:39:50 +08:00
createResults();
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
for (int i = 0; i < 2; i++)
{
int beforePanelCount = 0;
AddStep("get panel count", () => beforePanelCount = this.ChildrenOfType<ScorePanel>().Count());
AddStep("scroll to right", () => resultsScreen.ScorePanelList.ChildrenOfType<OsuScrollContainer>().Single().ScrollToEnd(false));
AddAssert("right loading spinner shown", () => resultsScreen.RightSpinner.State.Value == Visibility.Visible);
waitForDisplay();
AddAssert($"count increased by {scores_per_result}", () => this.ChildrenOfType<ScorePanel>().Count() >= beforePanelCount + scores_per_result);
2020-07-31 20:39:50 +08:00
AddAssert("right loading spinner hidden", () => resultsScreen.RightSpinner.State.Value == Visibility.Hidden);
}
}
[Test]
public void TestNoMoreScoresToTheRight()
{
AddStep("bind delayed handler with scores", () => bindHandler(delayed: true));
createResults();
waitForDisplay();
int beforePanelCount = 0;
AddStep("get panel count", () => beforePanelCount = this.ChildrenOfType<ScorePanel>().Count());
AddStep("scroll to right", () => resultsScreen.ScorePanelList.ChildrenOfType<OsuScrollContainer>().Single().ScrollToEnd(false));
AddAssert("right loading spinner shown", () => resultsScreen.RightSpinner.State.Value == Visibility.Visible);
waitForDisplay();
AddAssert($"count increased by {scores_per_result}", () => this.ChildrenOfType<ScorePanel>().Count() >= beforePanelCount + scores_per_result);
AddAssert("right loading spinner hidden", () => resultsScreen.RightSpinner.State.Value == Visibility.Hidden);
AddStep("get panel count", () => beforePanelCount = this.ChildrenOfType<ScorePanel>().Count());
AddStep("bind delayed handler with no scores", () => bindHandler(delayed: true, noScores: true));
AddStep("scroll to right", () => resultsScreen.ScorePanelList.ChildrenOfType<OsuScrollContainer>().Single().ScrollToEnd(false));
AddAssert("right loading spinner shown", () => resultsScreen.RightSpinner.State.Value == Visibility.Visible);
waitForDisplay();
AddAssert("count not increased", () => this.ChildrenOfType<ScorePanel>().Count() == beforePanelCount);
AddAssert("right loading spinner hidden", () => resultsScreen.RightSpinner.State.Value == Visibility.Hidden);
AddAssert("no placeholders shown", () => this.ChildrenOfType<MessagePlaceholder>().Count(), () => Is.Zero);
}
2020-07-31 20:39:50 +08:00
[Test]
public void TestFetchWhenScrolledToTheLeft()
{
AddStep("bind user score info handler", () => bindHandler(userScore: userScore));
2020-07-31 20:39:50 +08:00
createResults(() => userScore);
2024-01-09 20:28:46 +08:00
waitForDisplay();
2020-07-31 20:39:50 +08:00
AddStep("bind delayed handler", () => bindHandler(true));
2020-07-31 20:39:50 +08:00
for (int i = 0; i < 2; i++)
{
int beforePanelCount = 0;
AddStep("get panel count", () => beforePanelCount = this.ChildrenOfType<ScorePanel>().Count());
AddStep("scroll to left", () => resultsScreen.ScorePanelList.ChildrenOfType<OsuScrollContainer>().Single().ScrollToStart(false));
AddAssert("left loading spinner shown", () => resultsScreen.LeftSpinner.State.Value == Visibility.Visible);
waitForDisplay();
AddAssert($"count increased by {scores_per_result}", () => this.ChildrenOfType<ScorePanel>().Count() >= beforePanelCount + scores_per_result);
2020-07-31 20:39:50 +08:00
AddAssert("left loading spinner hidden", () => resultsScreen.LeftSpinner.State.Value == Visibility.Hidden);
}
}
[Test]
public void TestShowWithNoScores()
{
AddStep("bind user score info handler", () => bindHandler(noScores: true));
2024-01-09 20:28:46 +08:00
createResults();
AddAssert("no scores visible", () => !resultsScreen.ScorePanelList.GetScorePanels().Any());
AddAssert("placeholder shown", () => this.ChildrenOfType<MessagePlaceholder>().Count(), () => Is.EqualTo(1));
}
2024-01-09 20:28:46 +08:00
private void createResults(Func<ScoreInfo> getScore = null)
2020-06-09 17:53:55 +08:00
{
AddStep("load results", () =>
{
LoadScreen(resultsScreen = new TestResultsScreen(getScore?.Invoke(), 1, new PlaylistItem(new TestBeatmap(new OsuRuleset().RulesetInfo).BeatmapInfo)
2020-06-09 17:53:55 +08:00
{
RulesetID = new OsuRuleset().RulesetInfo.OnlineID
2020-06-09 17:53:55 +08:00
}));
});
2021-09-06 19:20:52 +08:00
AddUntilStep("wait for screen to load", () => resultsScreen.IsLoaded);
2020-06-09 17:53:55 +08:00
}
2024-01-09 20:28:46 +08:00
private void waitForDisplay()
2020-05-28 19:08:45 +08:00
{
AddUntilStep("wait for scores loaded", () =>
requestComplete
// request handler may need to fire more than once to get scores.
&& totalCount > 0
&& resultsScreen.ScorePanelList.GetScorePanels().Count() == totalCount
&& resultsScreen.ScorePanelList.AllPanelsVisible);
2020-07-31 20:39:50 +08:00
AddWaitStep("wait for display", 5);
}
private void bindHandler(bool delayed = false, ScoreInfo userScore = null, bool failRequests = false, bool noScores = false) => ((DummyAPIAccess)API).HandleRequest = request =>
2020-07-31 20:39:50 +08:00
{
// pre-check for requests we should be handling (as they are scheduled below).
switch (request)
{
2022-06-24 20:25:23 +08:00
case ShowPlaylistUserScoreRequest:
case IndexPlaylistScoresRequest:
break;
default:
return false;
}
2020-07-31 20:39:50 +08:00
requestComplete = false;
double delay = delayed ? 3000 : 0;
2020-07-31 20:39:50 +08:00
Scheduler.AddDelayed(() =>
2020-07-31 20:39:50 +08:00
{
if (failRequests)
{
triggerFail(request);
return;
}
switch (request)
{
case ShowPlaylistUserScoreRequest s:
if (userScore == null)
triggerFail(s);
else
triggerSuccess(s, createUserResponse(userScore));
break;
case IndexPlaylistScoresRequest i:
triggerSuccess(i, createIndexResponse(i, noScores));
break;
}
}, delay);
return true;
2020-07-31 20:39:50 +08:00
};
2020-05-28 19:08:45 +08:00
private void triggerSuccess<T>(APIRequest<T> req, T result)
2020-07-31 20:39:50 +08:00
where T : class
{
requestComplete = true;
req.TriggerSuccess(result);
2020-07-31 20:39:50 +08:00
}
private void triggerFail(APIRequest req)
2020-07-31 20:39:50 +08:00
{
requestComplete = true;
req.TriggerFailure(new WebException("Failed."));
2020-07-31 20:39:50 +08:00
}
private MultiplayerScore createUserResponse([NotNull] ScoreInfo userScore)
{
var multiplayerUserScore = new MultiplayerScore
{
ID = highestScoreId,
2020-07-31 20:39:50 +08:00
Accuracy = userScore.Accuracy,
Passed = userScore.Passed,
Rank = userScore.Rank,
Position = real_user_position,
2020-07-31 20:39:50 +08:00
MaxCombo = userScore.MaxCombo,
User = userScore.User,
ScoresAround = new MultiplayerScoresAround
{
Higher = new MultiplayerScores(),
Lower = new MultiplayerScores()
}
};
totalCount++;
2020-07-31 20:39:50 +08:00
for (int i = 1; i <= scores_per_result; i++)
{
multiplayerUserScore.ScoresAround.Lower.Scores.Add(new MultiplayerScore
{
ID = getNextLowestScoreId(),
2020-07-31 20:39:50 +08:00
Accuracy = userScore.Accuracy,
Passed = true,
Rank = userScore.Rank,
MaxCombo = userScore.MaxCombo,
User = new APIUser
2020-07-31 20:39:50 +08:00
{
Id = 2,
Username = $"peppy{i}",
CoverUrl = "https://osu.ppy.sh/images/headers/profile-covers/c3.jpg",
},
});
multiplayerUserScore.ScoresAround.Higher.Scores.Add(new MultiplayerScore
{
ID = getNextHighestScoreId(),
2020-07-31 20:39:50 +08:00
Accuracy = userScore.Accuracy,
Passed = true,
Rank = userScore.Rank,
MaxCombo = userScore.MaxCombo,
User = new APIUser
2020-07-31 20:39:50 +08:00
{
Id = 2,
Username = $"peppy{i}",
CoverUrl = "https://osu.ppy.sh/images/headers/profile-covers/c3.jpg",
},
});
totalCount += 2;
2020-07-31 20:39:50 +08:00
}
addCursor(multiplayerUserScore.ScoresAround.Lower);
addCursor(multiplayerUserScore.ScoresAround.Higher);
return multiplayerUserScore;
}
private IndexedMultiplayerScores createIndexResponse(IndexPlaylistScoresRequest req, bool noScores = false)
2020-07-31 20:39:50 +08:00
{
var result = new IndexedMultiplayerScores();
if (noScores) return result;
2020-07-31 20:39:50 +08:00
string sort = req.IndexParams?.Properties["sort"].ToObject<string>() ?? "score_desc";
for (int i = 1; i <= scores_per_result; i++)
{
result.Scores.Add(new MultiplayerScore
{
ID = sort == "score_asc" ? getNextHighestScoreId() : getNextLowestScoreId(),
2020-07-31 20:39:50 +08:00
Accuracy = 1,
2020-05-28 19:08:45 +08:00
Passed = true,
2020-07-31 20:39:50 +08:00
Rank = ScoreRank.X,
MaxCombo = 1000,
User = new APIUser
2020-05-28 19:08:45 +08:00
{
Id = 2,
Username = $"peppy{i}",
CoverUrl = "https://osu.ppy.sh/images/headers/profile-covers/c3.jpg",
},
});
totalCount++;
2020-05-28 19:08:45 +08:00
}
2020-07-31 20:39:50 +08:00
addCursor(result);
return result;
}
/// <summary>
/// The next highest score ID to appear at the left of the list. Monotonically decreasing.
/// </summary>
private int getNextHighestScoreId() => --highestScoreId;
/// <summary>
/// The next lowest score ID to appear at the right of the list. Monotonically increasing.
/// </summary>
/// <returns></returns>
private int getNextLowestScoreId() => ++lowestScoreId;
2020-07-31 20:39:50 +08:00
private void addCursor(MultiplayerScores scores)
{
scores.Cursor = new Cursor
{
Properties = new Dictionary<string, JToken>
{
{ "total_score", JToken.FromObject(scores.Scores[^1].TotalScore) },
{ "score_id", JToken.FromObject(scores.Scores[^1].ID) },
}
};
scores.Params = new IndexScoresParams
2020-05-28 19:08:45 +08:00
{
2020-07-31 20:39:50 +08:00
Properties = new Dictionary<string, JToken>
2020-05-28 19:08:45 +08:00
{
// [ 1, 2, 3, ... ] => score_desc (will be added to the right of the list)
// [ 3, 2, 1, ... ] => score_asc (will be added to the left of the list)
{ "sort", JToken.FromObject(scores.Scores[^1].ID > scores.Scores[^2].ID ? "score_desc" : "score_asc") }
2020-06-09 17:53:55 +08:00
}
};
2020-05-28 19:08:45 +08:00
}
2020-07-31 20:39:50 +08:00
2022-11-24 13:32:20 +08:00
private partial class TestResultsScreen : PlaylistsResultsScreen
2020-07-31 20:39:50 +08:00
{
public new LoadingSpinner LeftSpinner => base.LeftSpinner;
public new LoadingSpinner CentreSpinner => base.CentreSpinner;
public new LoadingSpinner RightSpinner => base.RightSpinner;
public new ScorePanelList ScorePanelList => base.ScorePanelList;
public TestResultsScreen([CanBeNull] ScoreInfo score, int roomId, PlaylistItem playlistItem)
: base(score, roomId, playlistItem)
2020-07-31 20:39:50 +08:00
{
AllowRetry = true;
2020-07-31 20:39:50 +08:00
}
}
2020-05-28 19:08:45 +08:00
}
}