osu/osu.Game.Tests/Visual/Gameplay/TestScenePlayerLoader.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

609 lines
23 KiB
C#
Raw Normal View History

// 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 07:37:17 +00:00
#nullable disable
using System;
2019-05-10 09:21:07 +00:00
using System.Collections.Generic;
2019-04-09 06:05:03 +00:00
using System.Linq;
using System.Threading;
2019-12-06 04:47:34 +00:00
using System.Threading.Tasks;
2019-04-09 04:50:54 +00:00
using NUnit.Framework;
using osu.Framework.Allocation;
2019-09-25 10:24:05 +00:00
using osu.Framework.Audio;
2019-02-12 10:53:08 +00:00
using osu.Framework.Graphics;
2019-09-25 10:24:05 +00:00
using osu.Framework.Graphics.Containers;
2019-01-23 11:52:00 +00:00
using osu.Framework.Screens;
2020-10-19 22:08:26 +00:00
using osu.Framework.Testing;
using osu.Framework.Utils;
using osu.Game.Beatmaps;
2019-10-01 16:15:40 +00:00
using osu.Game.Configuration;
2019-09-25 10:24:05 +00:00
using osu.Game.Overlays;
using osu.Game.Overlays.Notifications;
2019-04-09 06:05:03 +00:00
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Osu;
2019-12-06 04:47:34 +00:00
using osu.Game.Rulesets.Osu.Mods;
2019-04-09 06:05:03 +00:00
using osu.Game.Rulesets.Scoring;
using osu.Game.Scoring;
2024-02-01 22:17:28 +00:00
using osu.Game.Screens.Menu;
using osu.Game.Screens.Play;
2019-07-05 06:50:31 +00:00
using osu.Game.Screens.Play.PlayerSettings;
using osu.Game.Utils;
2024-02-01 22:17:28 +00:00
using osuTK;
2019-09-25 10:24:05 +00:00
using osuTK.Input;
2019-03-24 16:02:36 +00:00
namespace osu.Game.Tests.Visual.Gameplay
{
public partial class TestScenePlayerLoader : ScreenTestScene
{
2019-07-05 06:50:31 +00:00
private TestPlayerLoader loader;
2019-09-25 10:24:05 +00:00
private TestPlayer player;
2024-03-15 08:46:18 +00:00
private bool? epilepsyWarning;
private BeatmapOnlineStatus? onlineStatus;
2020-07-21 08:48:11 +00:00
2019-09-25 10:24:05 +00:00
[Resolved]
private AudioManager audioManager { get; set; }
2019-10-01 16:15:40 +00:00
[Resolved]
private SessionStatics sessionStatics { get; set; }
[Resolved]
private OsuConfigManager config { get; set; }
[Cached(typeof(INotificationOverlay))]
private readonly NotificationOverlay notificationOverlay;
[Cached]
private readonly VolumeOverlay volumeOverlay;
2024-02-01 22:17:28 +00:00
[Cached]
private readonly OsuLogo logo;
[Cached(typeof(BatteryInfo))]
private readonly LocalBatteryInfo batteryInfo = new LocalBatteryInfo();
private readonly ChangelogOverlay changelogOverlay;
private double savedTrackVolume;
private double savedMasterVolume;
private bool savedMutedState;
public TestScenePlayerLoader()
{
AddRange(new Drawable[]
{
notificationOverlay = new NotificationOverlay
{
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
},
volumeOverlay = new VolumeOverlay
{
Anchor = Anchor.TopLeft,
Origin = Anchor.TopLeft,
},
2024-02-01 22:17:28 +00:00
changelogOverlay = new ChangelogOverlay(),
logo = new OsuLogo
{
Anchor = Anchor.BottomRight,
Origin = Anchor.BottomRight,
Scale = new Vector2(0.5f),
Position = new Vector2(128f),
},
});
}
[SetUp]
2024-03-15 08:46:18 +00:00
public void Setup() => Schedule(() =>
{
player = null;
epilepsyWarning = null;
onlineStatus = null;
});
[SetUpSteps]
public override void SetUpSteps()
{
base.SetUpSteps();
AddStep("read all notifications", () =>
{
notificationOverlay.Show();
notificationOverlay.Hide();
});
AddUntilStep("wait for no notifications", () => notificationOverlay.UnreadCount.Value, () => Is.EqualTo(0));
}
2019-09-25 10:24:05 +00:00
/// <summary>
/// Sets the input manager child to a new test player loader container instance.
/// </summary>
/// <param name="interactive">If the test player should behave like the production one.</param>
/// <param name="beforeLoadAction">An action to run before player load but after bindable leases are returned.</param>
private void resetPlayer(bool interactive, Action beforeLoadAction = null)
{
2019-09-25 10:24:05 +00:00
beforeLoadAction?.Invoke();
prepareBeatmap();
LoadScreen(loader = new TestPlayerLoader(() => player = new TestPlayer(interactive, interactive)));
}
private void prepareBeatmap()
{
var workingBeatmap = CreateWorkingBeatmap(new OsuRuleset().RulesetInfo);
2022-08-16 05:17:28 +00:00
// Add intro time to test quick retry skipping (TestQuickRetry).
workingBeatmap.BeatmapInfo.AudioLeadIn = 60000;
// Set up data for testing disclaimer display.
2024-03-15 08:46:18 +00:00
workingBeatmap.BeatmapInfo.EpilepsyWarning = epilepsyWarning ?? false;
workingBeatmap.BeatmapInfo.Status = onlineStatus ?? BeatmapOnlineStatus.Ranked;
2022-08-16 05:17:28 +00:00
Beatmap.Value = workingBeatmap;
2019-09-25 10:24:05 +00:00
2019-12-13 12:45:38 +00:00
foreach (var mod in SelectedMods.Value.OfType<IApplicableToTrack>())
mod.ApplyToTrack(Beatmap.Value.Track);
2019-09-25 10:24:05 +00:00
}
[Test]
public void TestEarlyExitBeforePlayerConstruction()
{
AddStep("load dummy beatmap", () => resetPlayer(false, () => SelectedMods.Value = new[] { new OsuModNightcore() }));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddStep("exit loader", () => loader.Exit());
AddUntilStep("wait for not current", () => !loader.IsCurrentScreen());
AddAssert("player did not load", () => player == null);
AddUntilStep("player disposed", () => loader.DisposalTask == null);
AddAssert("mod rate still applied", () => Beatmap.Value.Track.Rate != 1);
2019-09-25 10:24:05 +00:00
}
2019-12-06 04:47:34 +00:00
/// <summary>
/// When <see cref="PlayerLoader"/> exits early, it has to wait for the player load task
/// to complete before running disposal on player. This previously caused an issue where mod
/// speed adjustments were undone too late, causing cross-screen pollution.
/// </summary>
[Test]
public void TestEarlyExitAfterPlayerConstruction()
2019-12-06 04:47:34 +00:00
{
AddStep("load dummy beatmap", () => resetPlayer(false, () => SelectedMods.Value = new[] { new OsuModNightcore() }));
2019-12-06 04:47:34 +00:00
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddAssert("mod rate applied", () => Beatmap.Value.Track.Rate != 1);
AddUntilStep("wait for non-null player", () => player != null);
2019-12-06 04:47:34 +00:00
AddStep("exit loader", () => loader.Exit());
AddUntilStep("wait for not current", () => !loader.IsCurrentScreen());
AddAssert("player did not load", () => !player.IsLoaded);
AddUntilStep("player disposed", () => loader.DisposalTask?.IsCompleted == true);
AddAssert("mod rate still applied", () => Beatmap.Value.Track.Rate != 1);
}
2019-07-05 06:50:31 +00:00
[Test]
public void TestBlockLoadViaMouseMovement()
{
AddStep("load dummy beatmap", () => resetPlayer(false));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
2020-02-14 10:02:11 +00:00
AddUntilStep("wait for load ready", () =>
{
moveMouse();
return player?.LoadState == LoadState.Ready;
2020-02-14 10:02:11 +00:00
});
2020-02-14 10:02:11 +00:00
AddRepeatStep("move mouse", moveMouse, 20);
AddAssert("loader still active", () => loader.IsCurrentScreen());
AddUntilStep("loads after idle", () => !loader.IsCurrentScreen());
void moveMouse()
{
notificationOverlay.State.Value = Visibility.Hidden;
2020-02-14 10:02:11 +00:00
InputManager.MoveMouseTo(
loader.VisualSettings.ScreenSpaceDrawQuad.TopLeft
+ (loader.VisualSettings.ScreenSpaceDrawQuad.BottomRight - loader.VisualSettings.ScreenSpaceDrawQuad.TopLeft)
* RNG.NextSingle());
}
}
[Test]
public void TestBlockLoadViaFocus()
{
AddStep("load dummy beatmap", () => resetPlayer(false));
2020-02-14 10:02:11 +00:00
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddStep("show focused overlay", () => changelogOverlay.Show());
AddUntilStep("overlay visible", () => changelogOverlay.IsPresent);
2020-02-14 10:02:11 +00:00
AddUntilStep("wait for load ready", () => player?.LoadState == LoadState.Ready);
2020-02-14 10:02:11 +00:00
AddRepeatStep("twiddle thumbs", () => { }, 20);
2019-07-05 06:50:31 +00:00
AddAssert("loader still active", () => loader.IsCurrentScreen());
2020-02-14 10:02:11 +00:00
AddStep("hide overlay", () => changelogOverlay.Hide());
2019-07-05 06:50:31 +00:00
AddUntilStep("loads after idle", () => !loader.IsCurrentScreen());
}
2024-02-01 22:17:28 +00:00
[Test]
public void TestLoadNotBlockedOnOsuLogo()
{
AddStep("load dummy beatmap", () => resetPlayer(false));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddUntilStep("wait for load ready", () =>
{
moveMouse();
return player?.LoadState == LoadState.Ready;
});
// move mouse in logo while waiting for load to still proceed (it shouldn't be blocked when hovering logo).
AddUntilStep("move mouse in logo", () =>
{
moveMouse();
return !loader.IsCurrentScreen();
});
void moveMouse()
{
notificationOverlay.State.Value = Visibility.Hidden;
InputManager.MoveMouseTo(
logo.ScreenSpaceDrawQuad.TopLeft
+ (logo.ScreenSpaceDrawQuad.BottomRight - logo.ScreenSpaceDrawQuad.TopLeft)
* RNG.NextSingle(0.3f, 0.7f));
}
}
2019-04-09 04:50:54 +00:00
[Test]
public void TestLoadContinuation()
{
SlowLoadPlayer slowPlayer = null;
AddStep("load slow dummy beatmap", () =>
{
prepareBeatmap();
slowPlayer = new SlowLoadPlayer(false, false);
LoadScreen(loader = new TestPlayerLoader(() => slowPlayer));
});
AddStep("schedule slow load", () => Scheduler.AddDelayed(() => slowPlayer.AllowLoad.Set(), 5000));
AddUntilStep("wait for player to be current", () => slowPlayer.IsCurrentScreen());
}
2019-04-09 06:05:03 +00:00
[Test]
public void TestModReinstantiation()
{
TestMod gameMod = null;
TestMod playerMod1 = null;
TestMod playerMod2 = null;
AddStep("load player", () => { resetPlayer(true, () => SelectedMods.Value = new[] { gameMod = new TestMod() }); });
2019-04-09 06:05:03 +00:00
AddUntilStep("wait for loader to become current", () => loader.IsCurrentScreen());
AddStep("mouse in centre", () => InputManager.MoveMouseTo(loader.ScreenSpaceDrawQuad.Centre));
AddUntilStep("wait for player to be current", () => player.IsCurrentScreen());
AddStep("retrieve mods", () => playerMod1 = (TestMod)player.GameplayState.Mods.Single());
2019-04-09 06:05:03 +00:00
AddAssert("game mods not applied", () => gameMod.Applied == false);
AddAssert("player mods applied", () => playerMod1.Applied);
AddStep("restart player", () =>
{
var lastPlayer = player;
2019-04-09 06:05:03 +00:00
player = null;
lastPlayer.Restart();
2019-04-09 06:05:03 +00:00
});
AddUntilStep("wait for player to be current", () => player.IsCurrentScreen());
AddStep("retrieve mods", () => playerMod2 = (TestMod)player.GameplayState.Mods.Single());
2019-04-09 06:05:03 +00:00
AddAssert("game mods not applied", () => gameMod.Applied == false);
AddAssert("player has different mods", () => playerMod1 != playerMod2);
AddAssert("player mods applied", () => playerMod2.Applied);
}
[Test]
public void TestModDisplayChanges()
{
var testMod = new TestMod();
AddStep("load player", () => resetPlayer(true));
AddUntilStep("wait for loader to become current", () => loader.IsCurrentScreen());
AddStep("set test mod in loader", () => loader.Mods.Value = new[] { testMod });
AddAssert("test mod is displayed", () => (TestMod)loader.DisplayedMods.Single() == testMod);
}
2019-09-25 10:24:05 +00:00
[Test]
2024-02-16 07:52:27 +00:00
public void TestMutedNotificationLowMusicVolume()
2020-02-14 10:27:32 +00:00
{
addVolumeSteps("master and music volumes", () =>
{
audioManager.Volume.Value = 0.6;
2024-02-15 23:56:09 +00:00
audioManager.VolumeTrack.Value = 0.01;
}, () => Precision.AlmostEquals(audioManager.Volume.Value, 0.6) && Precision.AlmostEquals(audioManager.VolumeTrack.Value, 0.5));
2020-02-14 10:27:32 +00:00
}
[Test]
public void TestMutedNotificationLowMasterVolume()
2020-02-14 10:27:32 +00:00
{
addVolumeSteps("master and music volumes", () =>
{
audioManager.Volume.Value = 0.01;
audioManager.VolumeTrack.Value = 0.6;
}, () => Precision.AlmostEquals(audioManager.Volume.Value, 0.5) && Precision.AlmostEquals(audioManager.VolumeTrack.Value, 0.6));
2020-02-14 10:27:32 +00:00
}
[Test]
2020-02-14 10:27:32 +00:00
public void TestMutedNotificationMuteButton()
{
addVolumeSteps("mute button", () =>
{
// Importantly, in the case the volume is muted but the user has a volume level set, it should be retained.
2024-02-16 07:52:27 +00:00
audioManager.Volume.Value = 0.5;
audioManager.VolumeTrack.Value = 0.5;
volumeOverlay.IsMuted.Value = true;
2024-02-16 07:52:27 +00:00
}, () => !volumeOverlay.IsMuted.Value && audioManager.Volume.Value == 0.5 && audioManager.VolumeTrack.Value == 0.5);
2020-02-14 10:27:32 +00:00
}
/// <remarks>
/// Created for avoiding copy pasting code for the same steps.
/// </remarks>
/// <param name="volumeName">What part of the volume system is checked</param>
/// <param name="beforeLoad">The action to be invoked to set the volume before loading</param>
/// <param name="assert">The function to be invoked and checked</param>
private void addVolumeSteps(string volumeName, Action beforeLoad, Func<bool> assert)
2019-09-25 10:24:05 +00:00
{
2019-10-01 16:15:40 +00:00
AddStep("reset notification lock", () => sessionStatics.GetBindable<bool>(Static.MutedAudioNotificationShownOnce).Value = false);
2019-09-25 10:24:05 +00:00
AddStep("load player", () => resetPlayer(false, beforeLoad));
AddUntilStep("wait for player", () => player?.LoadState == LoadState.Ready);
2019-09-25 10:24:05 +00:00
saveVolumes();
AddAssert("check for notification", () => notificationOverlay.UnreadCount.Value, () => Is.EqualTo(1));
2019-09-25 10:24:05 +00:00
clickNotification();
2019-09-25 10:24:05 +00:00
AddAssert("check " + volumeName, assert);
2020-02-14 10:27:32 +00:00
restoreVolumes();
2020-02-14 10:27:32 +00:00
AddUntilStep("wait for player load", () => player.IsLoaded);
2019-09-25 10:24:05 +00:00
}
2020-07-21 08:48:11 +00:00
[TestCase(true)]
[TestCase(false)]
public void TestEpilepsyWarning(bool warning)
{
saveVolumes();
setFullVolume();
AddStep("enable storyboards", () => config.SetValue(OsuSetting.ShowStoryboard, true));
2020-10-19 21:53:41 +00:00
AddStep("change epilepsy warning", () => epilepsyWarning = warning);
AddStep("load dummy beatmap", () => resetPlayer(false));
2020-10-19 22:08:26 +00:00
2020-07-21 08:48:11 +00:00
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
2020-10-19 22:08:26 +00:00
AddAssert($"epilepsy warning {(warning ? "present" : "absent")}", () => this.ChildrenOfType<PlayerLoaderDisclaimer>().Count(), () => Is.EqualTo(warning ? 1 : 0));
restoreVolumes();
}
[Test]
public void TestEpilepsyWarningWithDisabledStoryboard()
{
saveVolumes();
setFullVolume();
AddStep("disable storyboards", () => config.SetValue(OsuSetting.ShowStoryboard, false));
AddStep("change epilepsy warning", () => epilepsyWarning = true);
AddStep("load dummy beatmap", () => resetPlayer(false));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddUntilStep("epilepsy warning absent", () => this.ChildrenOfType<PlayerLoaderDisclaimer>().Single().Alpha, () => Is.Zero);
restoreVolumes();
2020-10-20 16:27:03 +00:00
}
[TestCase(BeatmapOnlineStatus.Loved, 1)]
[TestCase(BeatmapOnlineStatus.Qualified, 1)]
[TestCase(BeatmapOnlineStatus.Graveyard, 0)]
public void TestStatusWarning(BeatmapOnlineStatus status, int expectedDisclaimerCount)
{
saveVolumes();
setFullVolume();
AddStep("enable storyboards", () => config.SetValue(OsuSetting.ShowStoryboard, true));
AddStep("disable epilepsy warning", () => epilepsyWarning = false);
AddStep("set beatmap status", () => onlineStatus = status);
AddStep("load dummy beatmap", () => resetPlayer(false));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
AddAssert($"disclaimer count is {expectedDisclaimerCount}", () => this.ChildrenOfType<PlayerLoaderDisclaimer>().Count(), () => Is.EqualTo(expectedDisclaimerCount));
restoreVolumes();
}
[Test]
public void TestCombinedWarnings()
{
saveVolumes();
setFullVolume();
AddStep("enable storyboards", () => config.SetValue(OsuSetting.ShowStoryboard, true));
AddStep("disable epilepsy warning", () => epilepsyWarning = true);
AddStep("set beatmap status", () => onlineStatus = BeatmapOnlineStatus.Loved);
AddStep("load dummy beatmap", () => resetPlayer(false));
AddUntilStep("wait for current", () => loader.IsCurrentScreen());
2024-03-15 10:26:48 +00:00
AddAssert("disclaimer count is 2", () => this.ChildrenOfType<PlayerLoaderDisclaimer>().Count(), () => Is.EqualTo(2));
restoreVolumes();
}
2022-07-30 12:26:19 +00:00
[TestCase(true, 1.0, false)] // on battery, above cutoff --> no warning
[TestCase(false, 0.1, false)] // not on battery, below cutoff --> no warning
[TestCase(true, 0.25, true)] // on battery, at cutoff --> warning
[TestCase(true, null, false)] // on battery, level unknown --> no warning
public void TestLowBatteryNotification(bool onBattery, double? chargeLevel, bool shouldWarn)
{
AddStep("reset notification lock", () => sessionStatics.GetBindable<bool>(Static.LowBatteryNotificationShownOnce).Value = false);
// set charge status and level
2021-04-09 00:28:23 +00:00
AddStep("load player", () => resetPlayer(false, () =>
2021-04-09 00:38:16 +00:00
{
2022-07-30 12:26:19 +00:00
batteryInfo.SetOnBattery(onBattery);
batteryInfo.SetChargeLevel(chargeLevel);
2021-04-09 00:28:23 +00:00
}));
AddUntilStep("wait for player", () => player?.LoadState == LoadState.Ready);
if (shouldWarn)
clickNotification();
else
AddAssert("notification not triggered", () => notificationOverlay.UnreadCount.Value == 0);
AddUntilStep("wait for player load", () => player.IsLoaded);
}
private void restoreVolumes()
2020-10-20 16:27:03 +00:00
{
AddStep("restore previous volumes", () =>
{
audioManager.VolumeTrack.Value = savedTrackVolume;
audioManager.Volume.Value = savedMasterVolume;
volumeOverlay.IsMuted.Value = savedMutedState;
});
}
private void setFullVolume()
{
AddStep("set volumes to 100%", () =>
{
audioManager.VolumeTrack.Value = 1;
audioManager.Volume.Value = 1;
volumeOverlay.IsMuted.Value = false;
});
}
2020-10-20 16:27:03 +00:00
private void saveVolumes()
{
AddStep("save previous volumes", () =>
{
savedTrackVolume = audioManager.VolumeTrack.Value;
savedMasterVolume = audioManager.Volume.Value;
savedMutedState = volumeOverlay.IsMuted.Value;
});
2020-07-21 08:48:11 +00:00
}
2020-10-20 16:27:03 +00:00
[Test]
2022-08-08 19:12:38 +00:00
public void TestQuickRetry()
2020-10-20 16:27:03 +00:00
{
2022-08-16 05:17:28 +00:00
TestPlayer getCurrentPlayer() => loader.CurrentPlayer as TestPlayer;
bool checkSkipButtonVisible() => player.ChildrenOfType<SkipOverlay>().FirstOrDefault()?.IsButtonVisible == true;
TestPlayer previousPlayer = null;
AddStep("load dummy beatmap", () => resetPlayer(false));
2020-10-20 16:27:03 +00:00
2022-08-16 05:17:28 +00:00
AddUntilStep("wait for current", () => getCurrentPlayer()?.IsCurrentScreen() == true);
AddStep("store previous player", () => previousPlayer = getCurrentPlayer());
2020-10-20 16:27:03 +00:00
2022-08-16 05:17:28 +00:00
AddStep("Restart map normally", () => getCurrentPlayer().Restart());
AddUntilStep("wait for load", () => getCurrentPlayer()?.LoadedBeatmapSuccessfully == true);
2022-08-16 05:17:28 +00:00
AddUntilStep("restart completed", () => getCurrentPlayer() != null && getCurrentPlayer() != previousPlayer);
AddStep("store previous player", () => previousPlayer = getCurrentPlayer());
2020-10-20 16:27:03 +00:00
2022-08-16 05:17:28 +00:00
AddUntilStep("skip button visible", checkSkipButtonVisible);
AddStep("press quick retry key", () => InputManager.PressKey(Key.Tilde));
AddUntilStep("restart completed", () => getCurrentPlayer() != null && getCurrentPlayer() != previousPlayer);
AddStep("release quick retry key", () => InputManager.ReleaseKey(Key.Tilde));
2022-08-17 08:07:06 +00:00
AddUntilStep("wait for player", () => getCurrentPlayer()?.LoadState == LoadState.Ready);
2022-08-16 05:17:28 +00:00
AddUntilStep("time reached zero", () => getCurrentPlayer()?.GameplayClockContainer.CurrentTime > 0);
AddUntilStep("skip button not visible", () => !checkSkipButtonVisible());
2020-07-21 08:48:11 +00:00
}
private void clickNotification()
{
Notification notification = null;
AddUntilStep("wait for notification", () => (notification = notificationOverlay.ChildrenOfType<Notification>().FirstOrDefault()) != null);
AddStep("open notification overlay", () => notificationOverlay.Show());
AddStep("click notification", () => notification.TriggerClick());
}
2019-07-05 06:50:31 +00:00
private partial class TestPlayerLoader : PlayerLoader
{
public new VisualSettings VisualSettings => base.VisualSettings;
2019-12-06 04:47:34 +00:00
public new Task DisposalTask => base.DisposalTask;
public IReadOnlyList<Mod> DisplayedMods => MetadataInfo.Mods.Value;
2019-07-05 06:50:31 +00:00
public TestPlayerLoader(Func<Player> createPlayer)
: base(createPlayer)
{
}
}
private class TestMod : OsuModDoubleTime, IApplicableToScoreProcessor
2019-04-09 06:05:03 +00:00
{
public bool Applied { get; private set; }
public void ApplyToScoreProcessor(ScoreProcessor scoreProcessor)
{
Applied = true;
}
public ScoreRank AdjustRank(ScoreRank rank, double accuracy) => rank;
2019-04-09 06:05:03 +00:00
}
protected partial class SlowLoadPlayer : TestPlayer
{
public readonly ManualResetEventSlim AllowLoad = new ManualResetEventSlim(false);
public SlowLoadPlayer(bool allowPause = true, bool showResults = true)
: base(allowPause, showResults)
{
}
[BackgroundDependencyLoader]
private void load()
{
if (!AllowLoad.Wait(TimeSpan.FromSeconds(10)))
throw new TimeoutException();
}
}
/// <summary>
/// Mutable dummy BatteryInfo class for <see cref="TestScenePlayerLoader.TestLowBatteryNotification"/>
/// </summary>
/// <inheritdoc/>
private class LocalBatteryInfo : BatteryInfo
{
2022-07-30 12:26:19 +00:00
private bool onBattery;
private double? chargeLevel;
2022-07-30 12:26:19 +00:00
public override bool OnBattery => onBattery;
2022-07-30 12:26:19 +00:00
public override double? ChargeLevel => chargeLevel;
2022-07-30 12:26:19 +00:00
public void SetOnBattery(bool value)
{
2022-07-30 12:26:19 +00:00
onBattery = value;
}
2022-07-30 12:26:19 +00:00
public void SetChargeLevel(double? value)
{
chargeLevel = value;
}
}
}
}