1
0
mirror of https://github.com/ppy/osu.git synced 2024-09-22 04:07:25 +08:00
osu-lazer/osu.Game/Skinning/Skin.cs

183 lines
6.1 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.
2018-04-13 17:19:50 +08:00
using System;
2021-05-10 21:43:48 +08:00
using System.Collections.Generic;
using System.IO;
2021-05-10 21:43:48 +08:00
using System.Linq;
using System.Text;
using JetBrains.Annotations;
2021-05-10 21:43:48 +08:00
using Newtonsoft.Json;
2018-04-13 17:19:50 +08:00
using osu.Framework.Audio.Sample;
using osu.Framework.Bindables;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics;
2020-07-17 15:54:30 +08:00
using osu.Framework.Graphics.OpenGL.Textures;
2018-04-13 17:19:50 +08:00
using osu.Framework.Graphics.Textures;
using osu.Framework.Logging;
2019-08-23 19:32:43 +08:00
using osu.Game.Audio;
2021-11-29 17:02:09 +08:00
using osu.Game.Database;
2021-05-10 21:43:48 +08:00
using osu.Game.IO;
using osu.Game.Screens.Play.HUD;
2018-04-13 17:19:50 +08:00
namespace osu.Game.Skinning
{
2019-04-25 16:36:17 +08:00
public abstract class Skin : IDisposable, ISkin
2018-04-13 17:19:50 +08:00
{
/// <summary>
/// A texture store which can be used to perform user file loops for this skin.
/// </summary>
[CanBeNull]
protected TextureStore Textures { get; set; }
/// <summary>
/// A sample store which can be used to perform user file loops for this skin.
/// </summary>
[CanBeNull]
protected ISampleStore Samples { get; set; }
public readonly Live<SkinInfo> SkinInfo;
2018-04-13 17:19:50 +08:00
public SkinConfiguration Configuration { get; set; }
2018-04-13 17:19:50 +08:00
2021-05-10 21:43:48 +08:00
public IDictionary<SkinnableTarget, SkinnableInfo[]> DrawableComponentInfo => drawableComponentInfo;
private readonly Dictionary<SkinnableTarget, SkinnableInfo[]> drawableComponentInfo = new Dictionary<SkinnableTarget, SkinnableInfo[]>();
2018-04-13 17:19:50 +08:00
public abstract ISample GetSample(ISampleInfo sampleInfo);
2018-04-13 17:19:50 +08:00
2020-07-17 15:54:30 +08:00
public Texture GetTexture(string componentName) => GetTexture(componentName, default, default);
public abstract Texture GetTexture(string componentName, WrapMode wrapModeS, WrapMode wrapModeT);
2018-04-13 17:19:50 +08:00
public abstract IBindable<TValue> GetConfig<TLookup, TValue>(TLookup lookup);
2018-04-13 17:19:50 +08:00
private readonly RealmBackedResourceStore skinStorage;
protected Skin(SkinInfo skin, IStorageResourceProvider resources, [CanBeNull] Stream configurationStream = null)
2018-04-13 17:19:50 +08:00
{
if (resources.RealmAccess != null)
{
SkinInfo = skin.ToLive(resources.RealmAccess);
skinStorage = new RealmBackedResourceStore(skin, resources.Files);
var samples = resources?.AudioManager?.GetSampleStore(skinStorage);
if (samples != null)
samples.PlaybackConcurrency = OsuGameBase.SAMPLE_CONCURRENCY;
Samples = samples;
Textures = new TextureStore(resources?.CreateTextureLoaderStore(skinStorage));
skinStorage.AddExtension("ogg");
}
else
{
SkinInfo = skin.ToLiveUnmanaged();
}
configurationStream ??= skinStorage?.GetStream(@"skin.ini");
if (configurationStream != null)
2021-10-24 22:43:37 +08:00
// stream will be closed after use by LineBufferedReader.
ParseConfigurationStream(configurationStream);
else
Configuration = new SkinConfiguration();
2021-05-10 21:43:48 +08:00
2021-11-29 17:02:09 +08:00
// skininfo files may be null for default skin.
foreach (SkinnableTarget skinnableTarget in Enum.GetValues(typeof(SkinnableTarget)))
{
string filename = $"{skinnableTarget}.json";
byte[] bytes = skinStorage?.Get(filename);
if (bytes == null)
continue;
try
{
string jsonContent = Encoding.UTF8.GetString(bytes);
var deserializedContent = JsonConvert.DeserializeObject<IEnumerable<SkinnableInfo>>(jsonContent);
if (deserializedContent == null)
2021-11-29 17:02:09 +08:00
continue;
DrawableComponentInfo[skinnableTarget] = deserializedContent.ToArray();
}
catch (Exception ex)
{
Logger.Error(ex, "Failed to load skin configuration.");
}
}
}
protected virtual void ParseConfigurationStream(Stream stream)
{
using (LineBufferedReader reader = new LineBufferedReader(stream, true))
Configuration = new LegacySkinDecoder().Decode(reader);
}
2021-05-13 12:09:33 +08:00
/// <summary>
/// Remove all stored customisations for the provided target.
/// </summary>
/// <param name="targetContainer">The target container to reset.</param>
public void ResetDrawableTarget(ISkinnableTarget targetContainer)
{
DrawableComponentInfo.Remove(targetContainer.Target);
2021-05-10 21:43:48 +08:00
}
2021-05-13 12:09:33 +08:00
/// <summary>
/// Update serialised information for the provided target.
/// </summary>
/// <param name="targetContainer">The target container to serialise to this skin.</param>
public void UpdateDrawableTarget(ISkinnableTarget targetContainer)
2021-05-10 21:43:48 +08:00
{
DrawableComponentInfo[targetContainer.Target] = targetContainer.CreateSkinnableInfo().ToArray();
2021-05-10 21:43:48 +08:00
}
public virtual Drawable GetDrawableComponent(ISkinComponent component)
{
switch (component)
{
case SkinnableTargetComponent target:
2021-05-13 12:09:33 +08:00
if (!DrawableComponentInfo.TryGetValue(target.Target, out var skinnableInfo))
2021-05-10 21:43:48 +08:00
return null;
return new SkinnableTargetComponentsContainer
2021-05-10 21:43:48 +08:00
{
ChildrenEnumerable = skinnableInfo.Select(i => i.CreateInstance())
};
}
return null;
2018-04-13 17:19:50 +08:00
}
#region Disposal
~Skin()
{
// required to potentially clean up sample store from audio hierarchy.
2018-04-13 17:19:50 +08:00
Dispose(false);
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
private bool isDisposed;
protected virtual void Dispose(bool isDisposing)
{
if (isDisposed)
return;
2019-02-28 12:31:40 +08:00
2018-04-13 17:19:50 +08:00
isDisposed = true;
Textures?.Dispose();
Samples?.Dispose();
2018-04-13 17:19:50 +08:00
}
#endregion
}
}