2019-12-20 13:29:54 +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
|
|
|
|
|
2019-12-20 13:29:54 +08:00
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Linq;
|
2020-01-06 16:15:59 +08:00
|
|
|
using NUnit.Framework;
|
2019-12-20 13:29:54 +08:00
|
|
|
using osu.Framework.Allocation;
|
2020-08-12 23:29:23 +08:00
|
|
|
using osu.Framework.Audio;
|
2022-01-03 16:31:12 +08:00
|
|
|
using osu.Framework.Extensions;
|
2021-12-14 18:47:11 +08:00
|
|
|
using osu.Framework.Graphics;
|
2020-01-06 16:15:59 +08:00
|
|
|
using osu.Framework.Graphics.Cursor;
|
|
|
|
using osu.Framework.Platform;
|
|
|
|
using osu.Framework.Testing;
|
2020-01-10 01:38:03 +08:00
|
|
|
using osu.Framework.Utils;
|
2019-12-20 13:29:54 +08:00
|
|
|
using osu.Game.Beatmaps;
|
2020-12-07 17:00:45 +08:00
|
|
|
using osu.Game.Database;
|
2020-01-06 16:15:59 +08:00
|
|
|
using osu.Game.Graphics.Cursor;
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
2022-07-05 02:04:20 +08:00
|
|
|
using osu.Game.Models;
|
2021-11-04 17:02:44 +08:00
|
|
|
using osu.Game.Online.API.Requests.Responses;
|
2019-12-20 13:29:54 +08:00
|
|
|
using osu.Game.Online.Leaderboards;
|
|
|
|
using osu.Game.Overlays;
|
2020-01-06 16:15:59 +08:00
|
|
|
using osu.Game.Rulesets;
|
2022-01-10 13:36:21 +08:00
|
|
|
using osu.Game.Rulesets.Osu;
|
2019-12-20 13:29:54 +08:00
|
|
|
using osu.Game.Scoring;
|
|
|
|
using osu.Game.Screens.Select.Leaderboards;
|
2020-01-06 16:15:59 +08:00
|
|
|
using osu.Game.Tests.Resources;
|
2019-12-20 13:29:54 +08:00
|
|
|
using osuTK;
|
2020-01-06 16:15:59 +08:00
|
|
|
using osuTK.Input;
|
2019-12-20 13:29:54 +08:00
|
|
|
|
|
|
|
namespace osu.Game.Tests.Visual.UserInterface
|
|
|
|
{
|
2020-03-23 09:01:33 +08:00
|
|
|
public class TestSceneDeleteLocalScore : OsuManualInputManagerTestScene
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
private readonly ContextMenuContainer contextMenuContainer;
|
|
|
|
private readonly BeatmapLeaderboard leaderboard;
|
|
|
|
|
|
|
|
private RulesetStore rulesetStore;
|
|
|
|
private BeatmapManager beatmapManager;
|
|
|
|
private ScoreManager scoreManager;
|
|
|
|
|
2021-09-07 15:46:27 +08:00
|
|
|
private readonly List<ScoreInfo> importedScores = new List<ScoreInfo>();
|
2021-10-02 23:55:29 +08:00
|
|
|
|
|
|
|
private BeatmapInfo beatmapInfo;
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2022-04-18 17:09:14 +08:00
|
|
|
[Cached(typeof(IDialogOverlay))]
|
2019-12-20 13:57:14 +08:00
|
|
|
private readonly DialogOverlay dialogOverlay;
|
2019-12-20 13:29:54 +08:00
|
|
|
|
|
|
|
public TestSceneDeleteLocalScore()
|
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
Children = new Drawable[]
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
contextMenuContainer = new OsuContextMenuContainer
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
Child = leaderboard = new BeatmapLeaderboard
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
Origin = Anchor.Centre,
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
Size = new Vector2(550f, 450f),
|
|
|
|
Scope = BeatmapLeaderboardScope.Local,
|
2022-04-06 13:51:08 +08:00
|
|
|
BeatmapInfo = TestResources.CreateTestBeatmapSetInfo().Beatmaps.First()
|
2020-01-06 16:15:59 +08:00
|
|
|
}
|
2019-12-20 13:29:54 +08:00
|
|
|
},
|
2020-01-06 16:15:59 +08:00
|
|
|
dialogOverlay = new DialogOverlay()
|
|
|
|
};
|
2019-12-20 13:29:54 +08:00
|
|
|
}
|
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
protected override IReadOnlyDependencyContainer CreateChildDependencies(IReadOnlyDependencyContainer parent)
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
var dependencies = new DependencyContainer(base.CreateChildDependencies(parent));
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2022-02-16 16:13:51 +08:00
|
|
|
dependencies.Cache(rulesetStore = new RealmRulesetStore(Realm));
|
2022-01-25 11:58:15 +08:00
|
|
|
dependencies.Cache(beatmapManager = new BeatmapManager(LocalStorage, Realm, rulesetStore, null, dependencies.Get<AudioManager>(), Resources, dependencies.Get<GameHost>(), Beatmap.Default));
|
2022-07-12 02:45:39 +08:00
|
|
|
dependencies.Cache(scoreManager = new ScoreManager(dependencies.Get<RulesetStore>(), () => beatmapManager, LocalStorage, Realm, Scheduler, API));
|
2022-01-25 11:58:15 +08:00
|
|
|
Dependencies.Cache(Realm);
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2022-04-05 03:59:09 +08:00
|
|
|
return dependencies;
|
|
|
|
}
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
private void load() => Schedule(() =>
|
|
|
|
{
|
2022-01-10 13:36:21 +08:00
|
|
|
var imported = beatmapManager.Import(new ImportTask(TestResources.GetQuickTestBeatmapForImport())).GetResultSafely();
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2022-01-10 13:36:21 +08:00
|
|
|
imported?.PerformRead(s =>
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2022-01-12 14:34:39 +08:00
|
|
|
beatmapInfo = s.Beatmaps[0];
|
2022-01-10 13:36:21 +08:00
|
|
|
|
|
|
|
for (int i = 0; i < 50; i++)
|
2020-01-06 16:15:59 +08:00
|
|
|
{
|
2022-01-10 13:36:21 +08:00
|
|
|
var score = new ScoreInfo
|
|
|
|
{
|
|
|
|
OnlineID = i,
|
|
|
|
BeatmapInfo = beatmapInfo,
|
|
|
|
Accuracy = RNG.NextDouble(),
|
|
|
|
TotalScore = RNG.Next(1, 1000000),
|
|
|
|
MaxCombo = RNG.Next(1, 1000),
|
|
|
|
Rank = ScoreRank.XH,
|
|
|
|
User = new APIUser { Username = "TestUser" },
|
|
|
|
Ruleset = new OsuRuleset().RulesetInfo,
|
2022-07-05 02:04:20 +08:00
|
|
|
Files = { new RealmNamedFileUsage(new RealmFile { Hash = $"{i}" }, string.Empty) }
|
2022-01-10 13:36:21 +08:00
|
|
|
};
|
|
|
|
|
2022-01-25 14:23:51 +08:00
|
|
|
importedScores.Add(scoreManager.Import(score).Value);
|
2022-01-10 13:36:21 +08:00
|
|
|
}
|
|
|
|
});
|
2020-01-06 16:15:59 +08:00
|
|
|
});
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-07 12:06:30 +08:00
|
|
|
[SetUpSteps]
|
|
|
|
public void SetupSteps()
|
2020-01-06 16:15:59 +08:00
|
|
|
{
|
2022-04-05 03:59:09 +08:00
|
|
|
AddUntilStep("ensure scores imported", () => importedScores.Count == 50);
|
|
|
|
AddStep("undelete scores", () =>
|
|
|
|
{
|
|
|
|
Realm.Run(r =>
|
|
|
|
{
|
|
|
|
// Due to soft deletions, we can re-use deleted scores between test runs
|
|
|
|
scoreManager.Undelete(r.All<ScoreInfo>().Where(s => s.DeletePending).ToList());
|
|
|
|
});
|
|
|
|
});
|
|
|
|
AddStep("set up leaderboard", () =>
|
|
|
|
{
|
|
|
|
leaderboard.BeatmapInfo = beatmapInfo;
|
|
|
|
leaderboard.RefetchScores(); // Required in the case that the beatmap hasn't changed
|
|
|
|
});
|
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
// Ensure the leaderboard items have finished showing up
|
|
|
|
AddStep("finish transforms", () => leaderboard.FinishTransforms(true));
|
2022-01-29 22:12:57 +08:00
|
|
|
AddUntilStep("wait for drawables", () => leaderboard.ChildrenOfType<LeaderboardScore>().Any());
|
2020-01-07 12:06:30 +08:00
|
|
|
}
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-07 12:06:30 +08:00
|
|
|
[Test]
|
|
|
|
public void TestDeleteViaRightClick()
|
|
|
|
{
|
2021-09-07 15:46:27 +08:00
|
|
|
ScoreInfo scoreBeingDeleted = null;
|
2020-01-06 16:15:59 +08:00
|
|
|
AddStep("open menu for top score", () =>
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2021-09-07 15:46:27 +08:00
|
|
|
var leaderboardScore = leaderboard.ChildrenOfType<LeaderboardScore>().First();
|
|
|
|
|
|
|
|
scoreBeingDeleted = leaderboardScore.Score;
|
|
|
|
|
|
|
|
InputManager.MoveMouseTo(leaderboardScore);
|
2020-01-06 16:15:59 +08:00
|
|
|
InputManager.Click(MouseButton.Right);
|
|
|
|
});
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
// Ensure the context menu has finished showing
|
|
|
|
AddStep("finish transforms", () => contextMenuContainer.FinishTransforms(true));
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
AddStep("click delete option", () =>
|
|
|
|
{
|
2021-02-22 16:14:00 +08:00
|
|
|
InputManager.MoveMouseTo(contextMenuContainer.ChildrenOfType<DrawableOsuMenuItem>().First(i => i.Item.Text.Value.ToString().ToLowerInvariant() == "delete"));
|
2020-01-06 16:15:59 +08:00
|
|
|
InputManager.Click(MouseButton.Left);
|
|
|
|
});
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
// Ensure the dialog has finished showing
|
|
|
|
AddStep("finish transforms", () => dialogOverlay.FinishTransforms(true));
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
AddStep("click delete button", () =>
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2020-01-06 16:15:59 +08:00
|
|
|
InputManager.MoveMouseTo(dialogOverlay.ChildrenOfType<DialogButton>().First());
|
2022-04-05 04:44:35 +08:00
|
|
|
InputManager.PressButton(MouseButton.Left);
|
2020-01-06 16:15:59 +08:00
|
|
|
});
|
2019-12-20 13:29:54 +08:00
|
|
|
|
2021-10-05 17:17:20 +08:00
|
|
|
AddUntilStep("wait for fetch", () => leaderboard.Scores != null);
|
2021-12-10 14:28:41 +08:00
|
|
|
AddUntilStep("score removed from leaderboard", () => leaderboard.Scores.All(s => s.OnlineID != scoreBeingDeleted.OnlineID));
|
2022-04-05 05:02:07 +08:00
|
|
|
|
|
|
|
// "Clean up"
|
|
|
|
AddStep("release left mouse button", () => InputManager.ReleaseButton(MouseButton.Left));
|
2019-12-20 13:29:54 +08:00
|
|
|
}
|
|
|
|
|
2020-01-06 16:15:59 +08:00
|
|
|
[Test]
|
|
|
|
public void TestDeleteViaDatabase()
|
2019-12-20 13:29:54 +08:00
|
|
|
{
|
2021-09-07 15:46:27 +08:00
|
|
|
AddStep("delete top score", () => scoreManager.Delete(importedScores[0]));
|
2021-10-08 14:26:25 +08:00
|
|
|
AddUntilStep("wait for fetch", () => leaderboard.Scores != null);
|
2021-12-10 14:28:41 +08:00
|
|
|
AddUntilStep("score removed from leaderboard", () => leaderboard.Scores.All(s => s.OnlineID != importedScores[0].OnlineID));
|
2019-12-20 13:29:54 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|