2019-01-24 08:43:03 +00: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.
2018-04-13 09:19:50 +00:00
using System ;
using System.Collections.Generic ;
using System.Linq ;
using System.Linq.Expressions ;
2019-05-28 09:59:21 +00:00
using System.Threading ;
using System.Threading.Tasks ;
2021-11-29 08:15:48 +00:00
using JetBrains.Annotations ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Audio ;
using osu.Framework.Audio.Sample ;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics ;
2020-07-17 07:54:30 +00:00
using osu.Framework.Graphics.OpenGL.Textures ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics.Textures ;
2019-08-29 07:38:39 +00:00
using osu.Framework.IO.Stores ;
2018-04-13 09:19:50 +00:00
using osu.Framework.Platform ;
2020-10-16 05:39:02 +00:00
using osu.Framework.Testing ;
2021-11-29 09:07:32 +00:00
using osu.Framework.Threading ;
2020-11-11 04:05:03 +00:00
using osu.Framework.Utils ;
2019-08-23 11:32:43 +00:00
using osu.Game.Audio ;
2018-04-13 09:19:50 +00:00
using osu.Game.Database ;
2020-12-21 06:14:32 +00:00
using osu.Game.IO ;
2018-04-13 09:19:50 +00:00
using osu.Game.IO.Archives ;
2021-11-25 06:14:43 +00:00
using osu.Game.Overlays.Notifications ;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Skinning
{
2021-06-22 00:06:39 +00:00
/// <summary>
/// Handles the storage and retrieval of <see cref="Skin"/>s.
/// </summary>
/// <remarks>
2021-06-22 09:05:17 +00:00
/// This is also exposed and cached as <see cref="ISkinSource"/> to allow for any component to potentially have skinning support.
/// For gameplay components, see <see cref="RulesetSkinProvidingContainer"/> which adds extra legacy and toggle logic that may affect the lookup process.
2021-06-22 00:06:39 +00:00
/// </remarks>
2020-10-16 05:39:02 +00:00
[ExcludeFromDynamicCompile]
2021-11-29 09:07:32 +00:00
public class SkinManager : ISkinSource , IStorageResourceProvider , IModelImporter < SkinInfo >
2018-04-13 09:19:50 +00:00
{
private readonly AudioManager audio ;
2021-11-29 07:18:34 +00:00
private readonly Scheduler scheduler ;
2020-12-21 06:14:32 +00:00
private readonly GameHost host ;
2021-05-31 09:37:32 +00:00
private readonly IResourceStore < byte [ ] > resources ;
2019-08-29 07:38:39 +00:00
2021-05-27 02:48:48 +00:00
public readonly Bindable < Skin > CurrentSkin = new Bindable < Skin > ( ) ;
2021-11-29 08:15:26 +00:00
2021-12-14 05:21:23 +00:00
public readonly Bindable < ILive < SkinInfo > > CurrentSkinInfo = new Bindable < ILive < SkinInfo > > ( Skinning . DefaultSkin . CreateInfo ( ) . ToLiveUnmanaged ( ) )
2021-11-29 08:15:26 +00:00
{
2021-12-14 05:21:23 +00:00
Default = Skinning . DefaultSkin . CreateInfo ( ) . ToLiveUnmanaged ( )
2021-11-29 08:15:26 +00:00
} ;
2018-04-13 09:19:50 +00:00
2021-11-25 06:14:43 +00:00
private readonly SkinModelManager skinModelManager ;
2021-11-29 09:07:32 +00:00
private readonly RealmContextFactory contextFactory ;
2018-11-28 10:16:05 +00:00
2021-11-25 06:14:43 +00:00
private readonly IResourceStore < byte [ ] > userFiles ;
2018-08-31 09:28:53 +00:00
2021-06-10 10:04:34 +00:00
/// <summary>
2021-06-11 05:50:21 +00:00
/// The default skin.
2021-06-10 10:04:34 +00:00
/// </summary>
2021-06-11 05:49:35 +00:00
public Skin DefaultSkin { get ; }
2021-06-07 15:42:50 +00:00
2021-06-10 10:04:34 +00:00
/// <summary>
2021-06-11 05:50:21 +00:00
/// The default legacy skin.
2021-06-10 10:04:34 +00:00
/// </summary>
2021-06-11 05:49:35 +00:00
public Skin DefaultLegacySkin { get ; }
2021-06-09 09:51:34 +00:00
2021-11-29 09:07:32 +00:00
public SkinManager ( Storage storage , RealmContextFactory contextFactory , GameHost host , IResourceStore < byte [ ] > resources , AudioManager audio , Scheduler scheduler )
2018-11-28 10:01:22 +00:00
{
2021-11-29 09:07:32 +00:00
this . contextFactory = contextFactory ;
2018-11-28 10:01:22 +00:00
this . audio = audio ;
2021-11-29 07:18:34 +00:00
this . scheduler = scheduler ;
2020-12-21 06:14:32 +00:00
this . host = host ;
2021-05-31 09:37:32 +00:00
this . resources = resources ;
2018-11-28 10:01:22 +00:00
2021-11-29 09:07:32 +00:00
userFiles = new StorageBackedResourceStore ( storage . GetStorageForDirectory ( "files" ) ) ;
2021-11-25 06:14:43 +00:00
2021-11-29 09:07:32 +00:00
skinModelManager = new SkinModelManager ( storage , contextFactory , host , this ) ;
2021-11-25 06:14:43 +00:00
2021-11-29 08:15:26 +00:00
var defaultSkins = new [ ]
{
DefaultLegacySkin = new DefaultLegacySkin ( this ) ,
DefaultSkin = new DefaultSkin ( this ) ,
} ;
// Ensure the default entries are present.
using ( var context = contextFactory . CreateContext ( ) )
using ( var transaction = context . BeginWrite ( ) )
{
foreach ( var skin in defaultSkins )
{
if ( context . Find < SkinInfo > ( skin . SkinInfo . ID ) = = null )
context . Add ( skin . SkinInfo . Value ) ;
}
transaction . Commit ( ) ;
}
2021-06-07 15:42:50 +00:00
2021-11-29 09:07:32 +00:00
CurrentSkinInfo . ValueChanged + = skin = > CurrentSkin . Value = skin . NewValue . PerformRead ( GetSkin ) ;
2021-05-27 05:50:42 +00:00
2021-06-11 05:49:35 +00:00
CurrentSkin . Value = DefaultSkin ;
2018-11-28 10:01:22 +00:00
CurrentSkin . ValueChanged + = skin = >
{
2021-11-29 09:07:32 +00:00
if ( ! skin . NewValue . SkinInfo . Equals ( CurrentSkinInfo . Value ) )
2018-11-28 10:01:22 +00:00
throw new InvalidOperationException ( $"Setting {nameof(CurrentSkin)}'s value directly is not supported. Use {nameof(CurrentSkinInfo)} instead." ) ;
SourceChanged ? . Invoke ( ) ;
} ;
2018-04-13 09:19:50 +00:00
}
2020-11-11 04:05:03 +00:00
public void SelectRandomSkin ( )
{
2021-11-29 09:07:32 +00:00
using ( var context = contextFactory . CreateContext ( ) )
2020-11-11 04:05:03 +00:00
{
2021-11-29 09:07:32 +00:00
// choose from only user skins, removing the current selection to ensure a new one is chosen.
var randomChoices = context . All < SkinInfo > ( ) . Where ( s = > ! s . DeletePending & & s . ID ! = CurrentSkinInfo . Value . ID ) . ToArray ( ) ;
2020-11-11 04:05:03 +00:00
2021-11-29 09:07:32 +00:00
if ( randomChoices . Length = = 0 )
{
2021-12-14 05:21:23 +00:00
CurrentSkinInfo . Value = Skinning . DefaultSkin . CreateInfo ( ) . ToLiveUnmanaged ( ) ;
2021-11-29 09:07:32 +00:00
return ;
}
var chosen = randomChoices . ElementAt ( RNG . Next ( 0 , randomChoices . Length ) ) ;
2021-12-14 05:21:23 +00:00
CurrentSkinInfo . Value = chosen . ToLive ( contextFactory ) ;
2021-11-29 09:07:32 +00:00
}
2018-04-13 09:19:50 +00:00
}
/// <summary>
/// Retrieve a <see cref="Skin"/> instance for the provided <see cref="SkinInfo"/>
/// </summary>
/// <param name="skinInfo">The skin to lookup.</param>
/// <returns>A <see cref="Skin"/> instance correlating to the provided <see cref="SkinInfo"/>.</returns>
2021-05-31 09:58:40 +00:00
public Skin GetSkin ( SkinInfo skinInfo ) = > skinInfo . CreateInstance ( this ) ;
2021-05-11 08:00:56 +00:00
/// <summary>
/// Ensure that the current skin is in a state it can accept user modifications.
/// This will create a copy of any internal skin and being tracking in the database if not already.
/// </summary>
public void EnsureMutableSkin ( )
2018-04-13 09:19:50 +00:00
{
2021-11-29 09:07:32 +00:00
CurrentSkinInfo . Value . PerformRead ( s = >
2021-05-11 08:00:56 +00:00
{
2021-12-02 04:41:20 +00:00
if ( ! s . Protected )
2021-11-29 09:07:32 +00:00
return ;
// if the user is attempting to save one of the default skin implementations, create a copy first.
var result = skinModelManager . Import ( new SkinInfo
{
Name = s . Name + @" (modified)" ,
Creator = s . Creator ,
InstantiationInfo = s . InstantiationInfo ,
} ) . Result ;
if ( result ! = null )
2021-12-02 08:42:16 +00:00
{
// save once to ensure the required json content is populated.
// currently this only happens on save.
result . PerformRead ( skin = > Save ( skin . CreateInstance ( this ) ) ) ;
2021-11-29 09:07:32 +00:00
CurrentSkinInfo . Value = result ;
2021-12-02 08:42:16 +00:00
}
2021-11-29 09:07:32 +00:00
} ) ;
2018-04-13 09:19:50 +00:00
}
2021-05-10 13:43:48 +00:00
public void Save ( Skin skin )
{
2021-11-29 09:07:32 +00:00
if ( ! skin . SkinInfo . IsManaged )
2021-05-11 08:00:56 +00:00
throw new InvalidOperationException ( $"Attempting to save a skin which is not yet tracked. Call {nameof(EnsureMutableSkin)} first." ) ;
2021-05-10 13:43:48 +00:00
2021-11-29 09:07:32 +00:00
skinModelManager . Save ( skin ) ;
2021-05-10 13:43:48 +00:00
}
2018-04-13 09:19:50 +00:00
/// <summary>
/// Perform a lookup query on available <see cref="SkinInfo"/>s.
/// </summary>
/// <param name="query">The query.</param>
/// <returns>The first result for the provided query, or null if no results were found.</returns>
2021-11-29 09:07:32 +00:00
public ILive < SkinInfo > Query ( Expression < Func < SkinInfo , bool > > query )
{
using ( var context = contextFactory . CreateContext ( ) )
2021-12-14 05:21:23 +00:00
return context . All < SkinInfo > ( ) . FirstOrDefault ( query ) ? . ToLive ( contextFactory ) ;
2021-11-29 09:07:32 +00:00
}
2018-04-13 09:19:50 +00:00
public event Action SourceChanged ;
2021-06-06 03:17:55 +00:00
public Drawable GetDrawableComponent ( ISkinComponent component ) = > lookupWithFallback ( s = > s . GetDrawableComponent ( component ) ) ;
2018-04-13 09:19:50 +00:00
2021-06-06 03:17:55 +00:00
public Texture GetTexture ( string componentName , WrapMode wrapModeS , WrapMode wrapModeT ) = > lookupWithFallback ( s = > s . GetTexture ( componentName , wrapModeS , wrapModeT ) ) ;
2018-04-13 09:19:50 +00:00
2021-06-06 03:17:55 +00:00
public ISample GetSample ( ISampleInfo sampleInfo ) = > lookupWithFallback ( s = > s . GetSample ( sampleInfo ) ) ;
2018-04-13 09:19:50 +00:00
2021-06-06 03:17:55 +00:00
public IBindable < TValue > GetConfig < TLookup , TValue > ( TLookup lookup ) = > lookupWithFallback ( s = > s . GetConfig < TLookup , TValue > ( lookup ) ) ;
2020-12-21 06:14:32 +00:00
2021-06-07 15:42:50 +00:00
public ISkin FindProvider ( Func < ISkin , bool > lookupFunction )
{
2021-06-22 07:49:21 +00:00
foreach ( var source in AllSources )
{
if ( lookupFunction ( source ) )
return source ;
}
2021-06-09 09:51:34 +00:00
2021-06-07 15:42:50 +00:00
return null ;
}
2021-06-06 03:17:55 +00:00
2021-06-22 07:19:55 +00:00
public IEnumerable < ISkin > AllSources
{
get
{
yield return CurrentSkin . Value ;
2021-06-22 07:49:37 +00:00
if ( CurrentSkin . Value is LegacySkin & & CurrentSkin . Value ! = DefaultLegacySkin )
2021-06-22 07:19:55 +00:00
yield return DefaultLegacySkin ;
2021-06-22 07:49:37 +00:00
if ( CurrentSkin . Value ! = DefaultSkin )
yield return DefaultSkin ;
2021-06-22 07:19:55 +00:00
}
}
2021-06-09 09:51:34 +00:00
private T lookupWithFallback < T > ( Func < ISkin , T > lookupFunction )
2021-06-06 03:17:55 +00:00
where T : class
{
2021-06-22 07:19:55 +00:00
foreach ( var source in AllSources )
{
if ( lookupFunction ( source ) is T skinSourced )
return skinSourced ;
}
2021-06-06 03:17:55 +00:00
2021-06-22 07:19:55 +00:00
return null ;
2021-06-06 03:17:55 +00:00
}
2021-05-31 08:25:21 +00:00
2020-12-22 03:01:09 +00:00
#region IResourceStorageProvider
2020-12-21 06:14:32 +00:00
AudioManager IStorageResourceProvider . AudioManager = > audio ;
2021-05-31 09:37:32 +00:00
IResourceStore < byte [ ] > IStorageResourceProvider . Resources = > resources ;
2021-11-25 06:14:43 +00:00
IResourceStore < byte [ ] > IStorageResourceProvider . Files = > userFiles ;
2021-12-14 05:21:23 +00:00
RealmContextFactory IStorageResourceProvider . RealmContextFactory = > contextFactory ;
2020-12-21 06:14:32 +00:00
IResourceStore < TextureUpload > IStorageResourceProvider . CreateTextureLoaderStore ( IResourceStore < byte [ ] > underlyingStore ) = > host . CreateTextureLoaderStore ( underlyingStore ) ;
2020-12-22 03:01:09 +00:00
#endregion
2021-11-25 06:14:43 +00:00
#region Implementation of IModelImporter < SkinInfo >
public Action < Notification > PostNotification
{
set = > skinModelManager . PostNotification = value ;
}
public Action < IEnumerable < ILive < SkinInfo > > > PostImport
{
set = > skinModelManager . PostImport = value ;
}
public Task Import ( params string [ ] paths )
{
return skinModelManager . Import ( paths ) ;
}
public Task Import ( params ImportTask [ ] tasks )
{
return skinModelManager . Import ( tasks ) ;
}
public IEnumerable < string > HandledExtensions = > skinModelManager . HandledExtensions ;
public Task < IEnumerable < ILive < SkinInfo > > > Import ( ProgressNotification notification , params ImportTask [ ] tasks )
{
return skinModelManager . Import ( notification , tasks ) ;
}
public Task < ILive < SkinInfo > > Import ( ImportTask task , bool lowPriority = false , CancellationToken cancellationToken = default )
{
return skinModelManager . Import ( task , lowPriority , cancellationToken ) ;
}
public Task < ILive < SkinInfo > > Import ( ArchiveReader archive , bool lowPriority = false , CancellationToken cancellationToken = default )
{
return skinModelManager . Import ( archive , lowPriority , cancellationToken ) ;
}
public Task < ILive < SkinInfo > > Import ( SkinInfo item , ArchiveReader archive = null , bool lowPriority = false , CancellationToken cancellationToken = default )
{
return skinModelManager . Import ( item , archive , lowPriority , cancellationToken ) ;
}
#endregion
#region Implementation of IModelManager < SkinInfo >
2021-11-29 07:18:34 +00:00
public void Delete ( [ CanBeNull ] Expression < Func < SkinInfo , bool > > filter = null , bool silent = false )
2021-11-25 06:14:43 +00:00
{
2021-11-29 09:07:32 +00:00
using ( var context = contextFactory . CreateContext ( ) )
{
2021-11-29 08:15:48 +00:00
var items = context . All < SkinInfo > ( )
. Where ( s = > ! s . Protected & & ! s . DeletePending ) ;
if ( filter ! = null )
items = items . Where ( filter ) ;
2021-11-29 07:18:34 +00:00
// check the removed skin is not the current user choice. if it is, switch back to default.
Guid currentUserSkin = CurrentSkinInfo . Value . ID ;
if ( items . Any ( s = > s . ID = = currentUserSkin ) )
2021-12-14 05:21:23 +00:00
scheduler . Add ( ( ) = > CurrentSkinInfo . Value = Skinning . DefaultSkin . CreateInfo ( ) . ToLiveUnmanaged ( ) ) ;
2021-11-29 07:18:34 +00:00
2021-11-29 08:15:48 +00:00
skinModelManager . Delete ( items . ToList ( ) , silent ) ;
2021-11-29 09:07:32 +00:00
}
2021-11-25 06:14:43 +00:00
}
#endregion
2018-04-13 09:19:50 +00:00
}
}