diff --git a/osu.Android.props b/osu.Android.props
index b3842a528d..bcd5f9bd9a 100644
--- a/osu.Android.props
+++ b/osu.Android.props
@@ -52,6 +52,6 @@
-
+
diff --git a/osu.Game.Rulesets.Catch/CatchRuleset.cs b/osu.Game.Rulesets.Catch/CatchRuleset.cs
index e3c457693e..23ce444560 100644
--- a/osu.Game.Rulesets.Catch/CatchRuleset.cs
+++ b/osu.Game.Rulesets.Catch/CatchRuleset.cs
@@ -161,13 +161,13 @@ namespace osu.Game.Rulesets.Catch
switch (result)
{
case HitResult.LargeTickHit:
- return "large droplet";
+ return "Large droplet";
case HitResult.SmallTickHit:
- return "small droplet";
+ return "Small droplet";
case HitResult.LargeBonus:
- return "banana";
+ return "Banana";
}
return base.GetDisplayNameForHitResult(result);
diff --git a/osu.Game.Tests/Beatmaps/Formats/LegacyBeatmapEncoderTest.cs b/osu.Game.Tests/Beatmaps/Formats/LegacyBeatmapEncoderTest.cs
index a18f82fe4a..059432eeaf 100644
--- a/osu.Game.Tests/Beatmaps/Formats/LegacyBeatmapEncoderTest.cs
+++ b/osu.Game.Tests/Beatmaps/Formats/LegacyBeatmapEncoderTest.cs
@@ -169,6 +169,8 @@ namespace osu.Game.Tests.Beatmaps.Formats
protected override Track GetBeatmapTrack() => throw new NotImplementedException();
+ protected override ISkin GetSkin() => throw new NotImplementedException();
+
public override Stream GetStream(string storagePath) => throw new NotImplementedException();
}
}
diff --git a/osu.Game.Tests/Visual/Gameplay/TestSceneBeatmapSkinFallbacks.cs b/osu.Game.Tests/Visual/Gameplay/TestSceneBeatmapSkinFallbacks.cs
new file mode 100644
index 0000000000..d6bedffaa8
--- /dev/null
+++ b/osu.Game.Tests/Visual/Gameplay/TestSceneBeatmapSkinFallbacks.cs
@@ -0,0 +1,122 @@
+// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence.
+// See the LICENCE file in the repository root for full licence text.
+
+using System;
+using System.Linq;
+using osu.Framework.Allocation;
+using osu.Framework.Audio;
+using osu.Framework.Graphics.Containers;
+using osu.Framework.Lists;
+using osu.Framework.Testing;
+using osu.Framework.Timing;
+using osu.Framework.Utils;
+using osu.Game.Beatmaps;
+using osu.Game.Extensions;
+using osu.Game.Rulesets;
+using osu.Game.Rulesets.Osu;
+using osu.Game.Rulesets.Osu.Skinning.Legacy;
+using osu.Game.Rulesets.Scoring;
+using osu.Game.Screens.Play.HUD;
+using osu.Game.Skinning;
+using osu.Game.Storyboards;
+
+namespace osu.Game.Tests.Visual.Gameplay
+{
+ public class TestSceneBeatmapSkinFallbacks : OsuPlayerTestScene
+ {
+ private ISkin currentBeatmapSkin;
+
+ [Resolved]
+ private SkinManager skinManager { get; set; }
+
+ [Cached]
+ private ScoreProcessor scoreProcessor = new ScoreProcessor();
+
+ [Cached(typeof(HealthProcessor))]
+ private HealthProcessor healthProcessor = new DrainingHealthProcessor(0);
+
+ protected override bool HasCustomSteps => true;
+
+ protected void CreateSkinTest(SkinInfo gameCurrentSkin, Func getBeatmapSkin)
+ {
+ CreateTest(() =>
+ {
+ AddStep("setup skins", () =>
+ {
+ skinManager.CurrentSkinInfo.Value = gameCurrentSkin;
+ currentBeatmapSkin = getBeatmapSkin();
+ });
+ });
+ }
+
+ protected bool AssertComponentsFromExpectedSource(SkinnableTarget target, ISkin expectedSource)
+ {
+ var actualComponentsContainer = Player.ChildrenOfType().First(s => s.Target == target)
+ .ChildrenOfType().SingleOrDefault();
+
+ if (actualComponentsContainer == null)
+ return false;
+
+ var actualInfo = actualComponentsContainer.CreateSkinnableInfo();
+
+ var expectedComponentsContainer = (SkinnableTargetComponentsContainer)expectedSource.GetDrawableComponent(new SkinnableTargetComponent(target));
+ if (expectedComponentsContainer == null)
+ return false;
+
+ var expectedComponentsAdjustmentContainer = new Container
+ {
+ Position = actualComponentsContainer.Parent.ToSpaceOfOtherDrawable(actualComponentsContainer.DrawPosition, Content),
+ Size = actualComponentsContainer.DrawSize,
+ Child = expectedComponentsContainer,
+ };
+
+ Add(expectedComponentsAdjustmentContainer);
+ expectedComponentsAdjustmentContainer.UpdateSubTree();
+ var expectedInfo = expectedComponentsContainer.CreateSkinnableInfo();
+ Remove(expectedComponentsAdjustmentContainer);
+
+ return almostEqual(actualInfo, expectedInfo);
+
+ static bool almostEqual(SkinnableInfo info, SkinnableInfo other) =>
+ other != null
+ && info.Type == other.Type
+ && info.Anchor == other.Anchor
+ && info.Origin == other.Origin
+ && Precision.AlmostEquals(info.Position, other.Position)
+ && Precision.AlmostEquals(info.Scale, other.Scale)
+ && Precision.AlmostEquals(info.Rotation, other.Rotation)
+ && info.Children.SequenceEqual(other.Children, new FuncEqualityComparer(almostEqual));
+ }
+
+ protected override WorkingBeatmap CreateWorkingBeatmap(IBeatmap beatmap, Storyboard storyboard = null)
+ => new CustomSkinWorkingBeatmap(beatmap, storyboard, Clock, Audio, currentBeatmapSkin);
+
+ protected override Ruleset CreatePlayerRuleset() => new TestOsuRuleset();
+
+ private class CustomSkinWorkingBeatmap : ClockBackedTestWorkingBeatmap
+ {
+ private readonly ISkin beatmapSkin;
+
+ public CustomSkinWorkingBeatmap(IBeatmap beatmap, Storyboard storyboard, IFrameBasedClock referenceClock, AudioManager audio, ISkin beatmapSkin)
+ : base(beatmap, storyboard, referenceClock, audio)
+ {
+ this.beatmapSkin = beatmapSkin;
+ }
+
+ protected override ISkin GetSkin() => beatmapSkin;
+ }
+
+ private class TestOsuRuleset : OsuRuleset
+ {
+ public override ISkin CreateLegacySkinProvider(ISkinSource source, IBeatmap beatmap) => new TestOsuLegacySkinTransformer(source);
+
+ private class TestOsuLegacySkinTransformer : OsuLegacySkinTransformer
+ {
+ public TestOsuLegacySkinTransformer(ISkinSource source)
+ : base(source)
+ {
+ }
+ }
+ }
+ }
+}
diff --git a/osu.Game.Tests/WaveformTestBeatmap.cs b/osu.Game.Tests/WaveformTestBeatmap.cs
index cbed28641c..5477e4a0f8 100644
--- a/osu.Game.Tests/WaveformTestBeatmap.cs
+++ b/osu.Game.Tests/WaveformTestBeatmap.cs
@@ -11,6 +11,7 @@ using osu.Game.Beatmaps;
using osu.Game.IO.Archives;
using osu.Game.Rulesets;
using osu.Game.Rulesets.Osu;
+using osu.Game.Skinning;
using osu.Game.Tests.Beatmaps;
using osu.Game.Tests.Resources;
@@ -52,6 +53,8 @@ namespace osu.Game.Tests
protected override Waveform GetWaveform() => new Waveform(trackStore.GetStream(firstAudioFile));
+ protected override ISkin GetSkin() => null;
+
public override Stream GetStream(string storagePath) => null;
protected override Track GetBeatmapTrack() => trackStore.Get(firstAudioFile);
diff --git a/osu.Game/Beatmaps/BeatmapManager.cs b/osu.Game/Beatmaps/BeatmapManager.cs
index 5e975de77c..46e3a4f6d7 100644
--- a/osu.Game/Beatmaps/BeatmapManager.cs
+++ b/osu.Game/Beatmaps/BeatmapManager.cs
@@ -526,6 +526,7 @@ namespace osu.Game.Beatmaps
protected override IBeatmap GetBeatmap() => beatmap;
protected override Texture GetBackground() => null;
protected override Track GetBeatmapTrack() => null;
+ protected override ISkin GetSkin() => null;
public override Stream GetStream(string storagePath) => null;
}
}
diff --git a/osu.Game/Beatmaps/DummyWorkingBeatmap.cs b/osu.Game/Beatmaps/DummyWorkingBeatmap.cs
index 6922d1c286..ea7f45e53f 100644
--- a/osu.Game/Beatmaps/DummyWorkingBeatmap.cs
+++ b/osu.Game/Beatmaps/DummyWorkingBeatmap.cs
@@ -15,6 +15,7 @@ using osu.Game.Rulesets.Difficulty;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Objects;
using osu.Game.Rulesets.UI;
+using osu.Game.Skinning;
namespace osu.Game.Beatmaps
{
@@ -49,6 +50,8 @@ namespace osu.Game.Beatmaps
protected override Track GetBeatmapTrack() => GetVirtualTrack();
+ protected override ISkin GetSkin() => null;
+
public override Stream GetStream(string storagePath) => null;
private class DummyRulesetInfo : RulesetInfo
diff --git a/osu.Game/Beatmaps/WorkingBeatmap.cs b/osu.Game/Beatmaps/WorkingBeatmap.cs
index 73a167b7aa..712a3dd33a 100644
--- a/osu.Game/Beatmaps/WorkingBeatmap.cs
+++ b/osu.Game/Beatmaps/WorkingBeatmap.cs
@@ -324,7 +324,8 @@ namespace osu.Game.Beatmaps
public bool SkinLoaded => skin.IsResultAvailable;
public ISkin Skin => skin.Value;
- protected virtual ISkin GetSkin() => null;
+ protected abstract ISkin GetSkin();
+
private readonly RecyclableLazy skin;
public abstract Stream GetStream(string storagePath);
diff --git a/osu.Game/Screens/Edit/LegacyEditorBeatmapPatcher.cs b/osu.Game/Screens/Edit/LegacyEditorBeatmapPatcher.cs
index 66784fda54..6f92db98ee 100644
--- a/osu.Game/Screens/Edit/LegacyEditorBeatmapPatcher.cs
+++ b/osu.Game/Screens/Edit/LegacyEditorBeatmapPatcher.cs
@@ -11,6 +11,7 @@ using osu.Framework.Audio.Track;
using osu.Framework.Graphics.Textures;
using osu.Game.Beatmaps;
using osu.Game.IO;
+using osu.Game.Skinning;
using Decoder = osu.Game.Beatmaps.Formats.Decoder;
namespace osu.Game.Screens.Edit
@@ -117,6 +118,8 @@ namespace osu.Game.Screens.Edit
protected override Track GetBeatmapTrack() => throw new NotImplementedException();
+ protected override ISkin GetSkin() => throw new NotImplementedException();
+
public override Stream GetStream(string storagePath) => throw new NotImplementedException();
}
}
diff --git a/osu.Game/Tests/Beatmaps/BeatmapConversionTest.cs b/osu.Game/Tests/Beatmaps/BeatmapConversionTest.cs
index a97f6defe9..4935f7fc13 100644
--- a/osu.Game/Tests/Beatmaps/BeatmapConversionTest.cs
+++ b/osu.Game/Tests/Beatmaps/BeatmapConversionTest.cs
@@ -17,6 +17,7 @@ using osu.Game.IO;
using osu.Game.Rulesets;
using osu.Game.Rulesets.Mods;
using osu.Game.Rulesets.Objects;
+using osu.Game.Skinning;
namespace osu.Game.Tests.Beatmaps
{
@@ -216,6 +217,8 @@ namespace osu.Game.Tests.Beatmaps
protected override Track GetBeatmapTrack() => throw new NotImplementedException();
+ protected override ISkin GetSkin() => throw new NotImplementedException();
+
public override Stream GetStream(string storagePath) => throw new NotImplementedException();
protected override IBeatmapConverter CreateBeatmapConverter(IBeatmap beatmap, Ruleset ruleset)
diff --git a/osu.Game/Tests/Beatmaps/TestWorkingBeatmap.cs b/osu.Game/Tests/Beatmaps/TestWorkingBeatmap.cs
index 852006bc9b..bfce59c7de 100644
--- a/osu.Game/Tests/Beatmaps/TestWorkingBeatmap.cs
+++ b/osu.Game/Tests/Beatmaps/TestWorkingBeatmap.cs
@@ -6,6 +6,7 @@ using osu.Framework.Audio;
using osu.Framework.Audio.Track;
using osu.Framework.Graphics.Textures;
using osu.Game.Beatmaps;
+using osu.Game.Skinning;
using osu.Game.Storyboards;
namespace osu.Game.Tests.Beatmaps
@@ -36,6 +37,8 @@ namespace osu.Game.Tests.Beatmaps
protected override Storyboard GetStoryboard() => storyboard ?? base.GetStoryboard();
+ protected override ISkin GetSkin() => null;
+
public override Stream GetStream(string storagePath) => null;
protected override Texture GetBackground() => null;
diff --git a/osu.Game/osu.Game.csproj b/osu.Game/osu.Game.csproj
index fa2945db6a..5f0f3028f0 100644
--- a/osu.Game/osu.Game.csproj
+++ b/osu.Game/osu.Game.csproj
@@ -33,7 +33,7 @@
all
runtime; build; native; contentfiles; analyzers; buildtransitive
-
+
diff --git a/osu.iOS.props b/osu.iOS.props
index e35b1b5c42..59b026e0ad 100644
--- a/osu.iOS.props
+++ b/osu.iOS.props
@@ -70,7 +70,7 @@
-
+
@@ -93,7 +93,7 @@
-
+