1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-25 07:22:54 +08:00
osu-lazer/osu.Game/Skinning/SkinnableSprite.cs

152 lines
5.8 KiB
C#
Raw Normal View History

2019-06-24 13:39:20 +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-03-25 16:36:02 +08:00
using System;
2022-04-05 15:47:15 +08:00
using System.Collections.Generic;
using System.Linq;
2019-06-24 13:39:20 +08:00
using osu.Framework.Allocation;
using osu.Framework.Bindables;
using osu.Framework.Graphics;
using osu.Framework.Graphics.Containers;
2019-06-24 13:39:20 +08:00
using osu.Framework.Graphics.Sprites;
using osu.Framework.Graphics.Textures;
using osu.Game.Configuration;
using osu.Game.Graphics.Sprites;
using osu.Game.Localisation.SkinComponents;
using osu.Game.Overlays.Settings;
using osuTK;
2019-06-24 13:39:20 +08:00
namespace osu.Game.Skinning
{
2019-06-24 14:25:01 +08:00
/// <summary>
/// A skinnable element which uses a single texture backing.
2019-06-24 14:25:01 +08:00
/// </summary>
public partial class SkinnableSprite : SkinnableDrawable, ISerialisableDrawable
2019-06-24 13:39:20 +08:00
{
2019-06-24 14:25:01 +08:00
protected override bool ApplySizeRestrictionsToDefault => true;
2019-06-24 13:39:20 +08:00
[Resolved]
private TextureStore textures { get; set; } = null!;
2019-06-24 13:39:20 +08:00
[SettingSource(typeof(SkinnableComponentStrings), nameof(SkinnableComponentStrings.SpriteName), nameof(SkinnableComponentStrings.SpriteNameDescription), SettingControlType = typeof(SpriteSelectorControl))]
public Bindable<string> SpriteName { get; } = new Bindable<string>(string.Empty);
[Resolved]
private ISkinSource source { get; set; } = null!;
public SkinnableSprite(string textureName, Vector2? maxSize = null, ConfineMode confineMode = ConfineMode.NoScaling)
: base(new SpriteComponentLookup(textureName, maxSize), confineMode)
2019-06-24 13:39:20 +08:00
{
SpriteName.Value = textureName;
}
public SkinnableSprite()
: base(new SpriteComponentLookup(string.Empty), ConfineMode.NoScaling)
{
RelativeSizeAxes = Axes.None;
AutoSizeAxes = Axes.Both;
SpriteName.BindValueChanged(name =>
{
((SpriteComponentLookup)ComponentLookup).LookupName = name.NewValue ?? string.Empty;
if (IsLoaded)
SkinChanged(CurrentSkin);
});
2019-06-24 13:39:20 +08:00
}
2019-06-24 14:27:46 +08:00
protected override Drawable CreateDefault(ISkinComponentLookup lookup)
{
var spriteLookup = (SpriteComponentLookup)lookup;
var texture = textures.Get(spriteLookup.LookupName);
if (texture == null)
return new SpriteNotFound(spriteLookup.LookupName);
if (spriteLookup.MaxSize != null)
texture = texture.WithMaximumSize(spriteLookup.MaxSize.Value);
return new Sprite { Texture = texture };
}
public bool UsesFixedAnchor { get; set; }
internal class SpriteComponentLookup : ISkinComponentLookup
{
public string LookupName { get; set; }
public Vector2? MaxSize { get; set; }
public SpriteComponentLookup(string textureName, Vector2? maxSize = null)
{
2019-11-12 17:45:42 +08:00
LookupName = textureName;
MaxSize = maxSize;
}
}
2022-11-24 13:32:20 +08:00
public partial class SpriteSelectorControl : SettingsDropdown<string>
{
protected override void LoadComplete()
{
base.LoadComplete();
// Round-about way of getting the user's skin to find available resources.
// In the future we'll probably want to allow access to resources from the fallbacks, or potentially other skins
// but that requires further thought.
2022-04-05 15:47:15 +08:00
var highestPrioritySkin = getHighestPriorityUserSkin(((SkinnableSprite)SettingSourceObject).source.AllSources) as Skin;
string[]? availableFiles = highestPrioritySkin?.SkinInfo.PerformRead(s => s.Files
.Where(f => f.Filename.EndsWith(".png", StringComparison.Ordinal)
|| f.Filename.EndsWith(".jpg", StringComparison.Ordinal))
.Select(f => f.Filename).Distinct()).ToArray();
if (availableFiles?.Length > 0)
Items = availableFiles;
2022-04-05 15:47:15 +08:00
static ISkin? getHighestPriorityUserSkin(IEnumerable<ISkin> skins)
2022-04-05 15:47:15 +08:00
{
foreach (var skin in skins)
{
if (skin is ISkinTransformer transformer && isUserSkin(transformer.Skin))
2022-04-05 15:47:15 +08:00
return transformer.Skin;
if (isUserSkin(skin))
return skin;
}
return null;
}
// Temporarily used to exclude undesirable ISkin implementations
static bool isUserSkin(ISkin skin)
2022-09-17 23:14:49 +08:00
=> skin.GetType() == typeof(TrianglesSkin)
2022-12-12 14:52:29 +08:00
|| skin.GetType() == typeof(ArgonProSkin)
2022-09-15 15:02:57 +08:00
|| skin.GetType() == typeof(ArgonSkin)
2022-04-05 15:47:15 +08:00
|| skin.GetType() == typeof(DefaultLegacySkin)
|| skin.GetType() == typeof(LegacySkin);
}
}
2022-11-24 13:32:20 +08:00
public partial class SpriteNotFound : CompositeDrawable
{
public SpriteNotFound(string lookup)
{
AutoSizeAxes = Axes.Both;
InternalChildren = new Drawable[]
{
new SpriteIcon
{
Size = new Vector2(50),
Icon = FontAwesome.Solid.QuestionCircle
},
new OsuSpriteText
{
Position = new Vector2(25, 50),
Text = $"missing: {lookup}",
Origin = Anchor.TopCentre,
}
};
}
}
2019-06-24 13:39:20 +08:00
}
}