1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-23 04:02:55 +08:00
osu-lazer/osu.Game/Tests/Visual/SkinnableTestScene.cs

231 lines
8.4 KiB
C#
Raw Normal View History

2019-07-26 18:29:06 +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-07-26 18:29:06 +08:00
using System;
using System.Collections.Generic;
2019-08-20 16:39:16 +08:00
using System.Text.RegularExpressions;
using JetBrains.Annotations;
2019-07-26 18:29:06 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2022-08-02 18:50:57 +08:00
using osu.Framework.Graphics.Rendering;
using osu.Framework.Graphics.Shapes;
2019-08-20 16:39:16 +08:00
using osu.Framework.Graphics.Textures;
2019-07-26 18:29:06 +08:00
using osu.Framework.IO.Stores;
using osu.Framework.Platform;
using osu.Game.Beatmaps;
using osu.Game.Database;
using osu.Game.Graphics.Sprites;
using osu.Game.IO;
using osu.Game.Rulesets;
2019-07-26 18:29:06 +08:00
using osu.Game.Skinning;
using osuTK;
using osuTK.Graphics;
2019-07-26 18:29:06 +08:00
namespace osu.Game.Tests.Visual
2019-07-26 18:29:06 +08:00
{
2022-11-24 13:32:20 +08:00
public abstract partial class SkinnableTestScene : OsuGridTestScene, IStorageResourceProvider
2019-07-26 18:29:06 +08:00
{
2022-09-18 17:18:10 +08:00
private TrianglesSkin trianglesSkin;
2019-07-26 18:29:06 +08:00
private Skin metricsSkin;
2022-09-18 17:18:10 +08:00
private Skin legacySkin;
private Skin argonSkin;
2019-07-26 18:29:06 +08:00
private Skin specialSkin;
2019-12-12 21:27:11 +08:00
private Skin oldSkin;
2019-07-26 18:29:06 +08:00
[Resolved]
private GameHost host { get; set; }
2019-07-30 20:53:28 +08:00
protected SkinnableTestScene()
2019-12-12 21:27:11 +08:00
: base(2, 3)
2019-07-30 20:53:28 +08:00
{
}
2019-07-26 18:29:06 +08:00
[BackgroundDependencyLoader]
2022-01-15 08:06:39 +08:00
private void load()
2019-07-26 18:29:06 +08:00
{
2022-02-23 12:51:25 +08:00
var dllStore = new DllResourceStore(GetType().Assembly);
2019-07-26 18:29:06 +08:00
argonSkin = new ArgonSkin(this);
2022-09-18 17:18:10 +08:00
trianglesSkin = new TrianglesSkin(this);
metricsSkin = new TestLegacySkin(new SkinInfo { Name = "metrics-skin" }, new NamespacedResourceStore<byte[]>(dllStore, "Resources/metrics_skin"), this, true);
2022-09-18 17:18:10 +08:00
legacySkin = new DefaultLegacySkin(this);
specialSkin = new TestLegacySkin(new SkinInfo { Name = "special-skin" }, new NamespacedResourceStore<byte[]>(dllStore, "Resources/special_skin"), this, true);
oldSkin = new TestLegacySkin(new SkinInfo { Name = "old-skin" }, new NamespacedResourceStore<byte[]>(dllStore, "Resources/old_skin"), this, true);
2019-07-26 18:29:06 +08:00
}
private readonly List<Drawable> createdDrawables = new List<Drawable>();
protected void SetContents(Func<ISkin, Drawable> creationFunction)
2019-07-26 18:29:06 +08:00
{
createdDrawables.Clear();
var beatmap = CreateBeatmapForSkinProvider();
Cell(0).Child = createProvider(argonSkin, creationFunction, beatmap);
Cell(1).Child = createProvider(trianglesSkin, creationFunction, beatmap);
Cell(2).Child = createProvider(metricsSkin, creationFunction, beatmap);
Cell(3).Child = createProvider(legacySkin, creationFunction, beatmap);
Cell(4).Child = createProvider(specialSkin, creationFunction, beatmap);
Cell(5).Child = createProvider(oldSkin, creationFunction, beatmap);
2019-08-26 11:31:51 +08:00
}
protected IEnumerable<Drawable> CreatedDrawables => createdDrawables;
2021-06-01 15:13:56 +08:00
private Drawable createProvider(Skin skin, Func<ISkin, Drawable> creationFunction, IBeatmap beatmap)
2019-08-26 11:31:51 +08:00
{
2021-06-01 15:13:56 +08:00
var created = creationFunction(skin);
createdDrawables.Add(created);
2019-08-26 11:31:51 +08:00
Container childContainer;
OutlineBox outlineBox;
SkinProvidingContainer skinProvider;
ISkin provider = Ruleset.Value.CreateInstance().CreateSkinTransformer(skin, beatmap) ?? skin;
var children = new Container
{
RelativeSizeAxes = Axes.Both,
BorderColour = Color4.White,
BorderThickness = 3,
Masking = true,
Children = new Drawable[]
{
new Box
{
AlwaysPresent = true,
Alpha = 0,
RelativeSizeAxes = Axes.Both,
},
new OsuSpriteText
{
2022-09-15 15:02:57 +08:00
Text = skin.SkinInfo.Value.Name,
Scale = new Vector2(1.5f),
Padding = new MarginPadding(5),
},
childContainer = new Container
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Children = new Drawable[]
{
outlineBox = new OutlineBox(),
skinProvider = new SkinProvidingContainer(provider)
{
Child = created,
}
}
},
}
};
// run this once initially to bring things into a sane state as early as possible.
updateSizing();
// run this once after construction to handle the case the changes are made in a BDL/LoadComplete call.
Schedule(updateSizing);
return children;
void updateSizing()
{
bool autoSize = created.RelativeSizeAxes == Axes.None;
foreach (var c in new[] { childContainer, skinProvider })
{
2020-11-05 16:26:41 +08:00
c.RelativeSizeAxes = Axes.None;
c.AutoSizeAxes = Axes.None;
c.Size = Vector2.Zero;
2020-11-05 16:26:41 +08:00
if (autoSize)
c.AutoSizeAxes = Axes.Both;
else
{
c.RelativeSizeAxes = Axes.Both;
c.Anchor = Anchor.Centre;
c.Origin = Anchor.Centre;
c.Size = new Vector2(0.97f);
}
}
outlineBox.Alpha = autoSize ? 1 : 0;
}
}
/// <summary>
/// Creates the ruleset for adding the corresponding skin transforming component.
/// </summary>
[NotNull]
protected abstract Ruleset CreateRulesetForSkinProvider();
protected sealed override Ruleset CreateRuleset() => CreateRulesetForSkinProvider();
protected virtual IBeatmap CreateBeatmapForSkinProvider() => CreateWorkingBeatmap(Ruleset.Value).GetPlayableBeatmap(Ruleset.Value);
#region IResourceStorageProvider
2022-08-02 18:50:57 +08:00
public IRenderer Renderer => host.Renderer;
public AudioManager AudioManager => Audio;
2024-02-02 18:48:13 +08:00
public IResourceStore<byte[]> Files => null!;
public new IResourceStore<byte[]> Resources => base.Resources;
public IResourceStore<TextureUpload> CreateTextureLoaderStore(IResourceStore<byte[]> underlyingStore) => host.CreateTextureLoaderStore(underlyingStore);
2024-02-02 18:48:13 +08:00
RealmAccess IStorageResourceProvider.RealmAccess => null!;
#endregion
2022-11-24 13:32:20 +08:00
private partial class OutlineBox : CompositeDrawable
{
public OutlineBox()
{
BorderColour = Color4.IndianRed;
BorderThickness = 5;
Masking = true;
RelativeSizeAxes = Axes.Both;
InternalChild = new Box
2019-08-26 11:31:51 +08:00
{
RelativeSizeAxes = Axes.Both,
Alpha = 0,
Colour = Color4.Brown,
AlwaysPresent = true
};
}
2019-07-26 18:29:06 +08:00
}
2019-08-20 16:39:16 +08:00
private class TestLegacySkin : LegacySkin
2019-07-26 18:29:06 +08:00
{
2019-08-20 16:39:16 +08:00
private readonly bool extrapolateAnimations;
2019-07-30 20:53:28 +08:00
public TestLegacySkin(SkinInfo skin, IResourceStore<byte[]> fallbackStore, IStorageResourceProvider resources, bool extrapolateAnimations)
: base(skin, resources, fallbackStore)
2019-08-20 16:39:16 +08:00
{
this.extrapolateAnimations = extrapolateAnimations;
}
2019-07-30 20:53:28 +08:00
2020-07-17 15:54:30 +08:00
public override Texture GetTexture(string componentName, WrapMode wrapModeS, WrapMode wrapModeT)
2019-08-20 16:39:16 +08:00
{
var lookup = base.GetTexture(componentName, wrapModeS, wrapModeT);
2019-07-30 20:53:28 +08:00
if (lookup != null)
return lookup;
// extrapolate frames to test longer animations
if (extrapolateAnimations)
{
var match = Regex.Match(componentName, "-([0-9]*)");
if (match.Length > 0 && int.TryParse(match.Groups[1].Value, out int number) && number < 60)
return base.GetTexture(componentName.Replace($"-{number}", $"-{number % 2}"), wrapModeS, wrapModeT);
}
return null;
}
2019-07-26 18:29:06 +08:00
}
}
}