mirror of
https://github.com/Artemis-RGB/Artemis
synced 2025-12-13 05:48:35 +00:00
216 lines
7.5 KiB
C#
216 lines
7.5 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.Linq;
|
|
using System.Linq.Expressions;
|
|
using System.Reflection;
|
|
using System.Threading.Tasks;
|
|
using Artemis.Core.Exceptions;
|
|
using Artemis.Core.Models.Profile;
|
|
using Artemis.Core.Models.Surface;
|
|
using Artemis.Core.Plugins.DataModelExpansions;
|
|
using Artemis.Core.Plugins.DataModelExpansions.Attributes;
|
|
using Artemis.Core.Plugins.DataModelExpansions.Internal;
|
|
using Artemis.Core.Utilities;
|
|
using SkiaSharp;
|
|
|
|
namespace Artemis.Core.Plugins.Modules
|
|
{
|
|
/// <summary>
|
|
/// Allows you to add support for new games/applications while utilizing Artemis' profile engine and your own data
|
|
/// model
|
|
/// </summary>
|
|
public abstract class ProfileModule<T> : ProfileModule where T : DataModel
|
|
{
|
|
/// <summary>
|
|
/// The data model driving this module
|
|
/// </summary>
|
|
public T DataModel
|
|
{
|
|
get => (T) InternalDataModel;
|
|
internal set => InternalDataModel = value;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets whether this module must also expand the main data model
|
|
/// <para>
|
|
/// Note: If expanding the main data model is all you want your plugin to do, create a
|
|
/// <see cref="BaseDataModelExpansion" /> plugin instead.
|
|
/// </para>
|
|
/// </summary>
|
|
public bool ExpandsDataModel
|
|
{
|
|
get => InternalExpandsMainDataModel;
|
|
set => InternalExpandsMainDataModel = value;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Override to provide your own data model description. By default this returns a description matching your plugin
|
|
/// name and description
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public virtual DataModelPropertyAttribute GetDataModelDescription()
|
|
{
|
|
return new DataModelPropertyAttribute {Name = PluginInfo.Name, Description = PluginInfo.Description};
|
|
}
|
|
|
|
/// <summary>
|
|
/// Hide the provided property using a lambda expression, e.g. HideProperty(dm => dm.TimeDataModel.CurrentTimeUTC)
|
|
/// </summary>
|
|
/// <param name="propertyLambda">A lambda expression pointing to the property to ignore</param>
|
|
public void HideProperty<TProperty>(Expression<Func<T, TProperty>> propertyLambda)
|
|
{
|
|
var propertyInfo = ReflectionUtilities.GetPropertyInfo(DataModel, propertyLambda);
|
|
if (!HiddenPropertiesList.Any(p => p.Equals(propertyInfo)))
|
|
HiddenPropertiesList.Add(propertyInfo);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stop hiding the provided property using a lambda expression, e.g. ShowProperty(dm =>
|
|
/// dm.TimeDataModel.CurrentTimeUTC)
|
|
/// </summary>
|
|
/// <param name="propertyLambda">A lambda expression pointing to the property to stop ignoring</param>
|
|
public void ShowProperty<TProperty>(Expression<Func<T, TProperty>> propertyLambda)
|
|
{
|
|
var propertyInfo = ReflectionUtilities.GetPropertyInfo(DataModel, propertyLambda);
|
|
HiddenPropertiesList.RemoveAll(p => p.Equals(propertyInfo));
|
|
}
|
|
|
|
internal override void InternalEnablePlugin()
|
|
{
|
|
DataModel = Activator.CreateInstance<T>();
|
|
DataModel.PluginInfo = PluginInfo;
|
|
DataModel.DataModelDescription = GetDataModelDescription();
|
|
base.InternalEnablePlugin();
|
|
}
|
|
|
|
internal override void InternalDisablePlugin()
|
|
{
|
|
DataModel = null;
|
|
base.InternalDisablePlugin();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Allows you to add support for new games/applications while utilizing Artemis' profile engine
|
|
/// </summary>
|
|
public abstract class ProfileModule : Module
|
|
{
|
|
protected ProfileModule()
|
|
{
|
|
OpacityOverride = 1;
|
|
}
|
|
|
|
protected readonly List<PropertyInfo> HiddenPropertiesList = new List<PropertyInfo>();
|
|
|
|
/// <summary>
|
|
/// Gets a list of all properties ignored at runtime using IgnoreProperty(x => x.y)
|
|
/// </summary>
|
|
public ReadOnlyCollection<PropertyInfo> HiddenProperties => HiddenPropertiesList.AsReadOnly();
|
|
|
|
public Profile ActiveProfile { get; private set; }
|
|
|
|
/// <summary>
|
|
/// Disables updating the profile, rendering does continue
|
|
/// </summary>
|
|
public bool IsProfileUpdatingDisabled { get; set; }
|
|
|
|
/// <inheritdoc />
|
|
public override void Update(double deltaTime)
|
|
{
|
|
lock (this)
|
|
{
|
|
OpacityOverride = AnimatingProfileChange
|
|
? Math.Max(0, OpacityOverride - 0.1)
|
|
: Math.Min(1, OpacityOverride + 0.1);
|
|
|
|
// Update the profile
|
|
if (!IsProfileUpdatingDisabled)
|
|
ActiveProfile?.Update(deltaTime);
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public override void Render(double deltaTime, ArtemisSurface surface, SKCanvas canvas, SKImageInfo canvasInfo)
|
|
{
|
|
lock (this)
|
|
{
|
|
// Render the profile
|
|
ActiveProfile?.Render(deltaTime, canvas, canvasInfo);
|
|
}
|
|
}
|
|
|
|
internal async Task ChangeActiveProfileAnimated(Profile profile, ArtemisSurface surface)
|
|
{
|
|
if (profile != null && profile.Module != this)
|
|
throw new ArtemisCoreException($"Cannot activate a profile of module {profile.Module} on a module of plugin {PluginInfo}.");
|
|
if (!IsActivated)
|
|
throw new ArtemisCoreException("Cannot activate a profile on a deactivated module");
|
|
|
|
if (profile == ActiveProfile || AnimatingProfileChange)
|
|
return;
|
|
|
|
AnimatingProfileChange = true;
|
|
|
|
while (OpacityOverride > 0)
|
|
await Task.Delay(50);
|
|
|
|
ChangeActiveProfile(profile, surface);
|
|
AnimatingProfileChange = false;
|
|
|
|
while (OpacityOverride < 1)
|
|
await Task.Delay(50);
|
|
}
|
|
|
|
internal void ChangeActiveProfile(Profile profile, ArtemisSurface surface)
|
|
{
|
|
if (profile != null && profile.Module != this)
|
|
throw new ArtemisCoreException($"Cannot activate a profile of module {profile.Module} on a module of plugin {PluginInfo}.");
|
|
if (!IsActivated)
|
|
throw new ArtemisCoreException("Cannot activate a profile on a deactivated module");
|
|
|
|
lock (this)
|
|
{
|
|
if (profile == ActiveProfile)
|
|
return;
|
|
|
|
ActiveProfile?.Dispose();
|
|
|
|
ActiveProfile = profile;
|
|
ActiveProfile?.Activate(surface);
|
|
}
|
|
|
|
OnActiveProfileChanged();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Overrides the opacity of the root folder
|
|
/// </summary>
|
|
public double OpacityOverride { get; set; }
|
|
|
|
/// <summary>
|
|
/// Indicates whether or not a profile change is being animated
|
|
/// </summary>
|
|
public bool AnimatingProfileChange { get; private set; }
|
|
|
|
internal override void Deactivate(bool isOverride)
|
|
{
|
|
base.Deactivate(isOverride);
|
|
|
|
var profile = ActiveProfile;
|
|
ActiveProfile = null;
|
|
profile?.Dispose();
|
|
}
|
|
|
|
#region Events
|
|
|
|
public event EventHandler ActiveProfileChanged;
|
|
|
|
protected virtual void OnActiveProfileChanged()
|
|
{
|
|
ActiveProfileChanged?.Invoke(this, EventArgs.Empty);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
} |