1
0
mirror of https://github.com/Artemis-RGB/Artemis synced 2026-01-02 10:43:31 +00:00

Data model debug - Added semi support for lists

This commit is contained in:
Robert 2020-06-29 22:28:12 +02:00
parent 796c0dc671
commit c80e5e42aa
11 changed files with 210 additions and 74 deletions

View File

@ -99,6 +99,11 @@ namespace Artemis.Core.Services.Interfaces
/// </summary> /// </summary>
event EventHandler<PluginEventArgs> PluginUnloaded; event EventHandler<PluginEventArgs> PluginUnloaded;
/// <summary>
/// Occurs when a plugin is being enabled
/// </summary>
event EventHandler<PluginEventArgs> PluginEnabling;
/// <summary> /// <summary>
/// Occurs when a plugin has been enabled /// Occurs when a plugin has been enabled
/// </summary> /// </summary>

View File

@ -287,10 +287,11 @@ namespace Artemis.Core.Services
public void EnablePlugin(Plugin plugin, bool isAutoEnable = false) public void EnablePlugin(Plugin plugin, bool isAutoEnable = false)
{ {
_logger.Debug("Enabling plugin {pluginInfo}", plugin.PluginInfo);
OnPluginEnabling(new PluginEventArgs(plugin.PluginInfo));
lock (_plugins) lock (_plugins)
{ {
_logger.Debug("Enabling plugin {pluginInfo}", plugin.PluginInfo);
try try
{ {
plugin.SetEnabled(true, isAutoEnable); plugin.SetEnabled(true, isAutoEnable);
@ -402,6 +403,7 @@ namespace Artemis.Core.Services
public event EventHandler<PluginEventArgs> PluginLoading; public event EventHandler<PluginEventArgs> PluginLoading;
public event EventHandler<PluginEventArgs> PluginLoaded; public event EventHandler<PluginEventArgs> PluginLoaded;
public event EventHandler<PluginEventArgs> PluginUnloaded; public event EventHandler<PluginEventArgs> PluginUnloaded;
public event EventHandler<PluginEventArgs> PluginEnabling;
public event EventHandler<PluginEventArgs> PluginEnabled; public event EventHandler<PluginEventArgs> PluginEnabled;
public event EventHandler<PluginEventArgs> PluginDisabled; public event EventHandler<PluginEventArgs> PluginDisabled;
@ -425,6 +427,11 @@ namespace Artemis.Core.Services
PluginUnloaded?.Invoke(this, e); PluginUnloaded?.Invoke(this, e);
} }
protected virtual void OnPluginEnabling(PluginEventArgs e)
{
PluginEnabling?.Invoke(this, e);
}
protected virtual void OnPluginEnabled(PluginEventArgs e) protected virtual void OnPluginEnabled(PluginEventArgs e)
{ {
PluginEnabled?.Invoke(this, e); PluginEnabled?.Invoke(this, e);

View File

@ -1,15 +0,0 @@
using System.Reflection;
using Artemis.Core.Plugins.Abstract.DataModels.Attributes;
using Stylet;
namespace Artemis.UI.DataModelVisualization
{
public abstract class DataModelVisualizationViewModel : PropertyChangedBase
{
public PropertyInfo PropertyInfo { get; protected set; }
public DataModelPropertyAttribute PropertyDescription { get; protected set; }
public DataModelViewModel Parent { get; protected set; }
public abstract void Update();
}
}

View File

@ -0,0 +1,58 @@
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using Artemis.Core.Plugins.Abstract.DataModels.Attributes;
using Stylet;
namespace Artemis.UI.DataModelVisualization
{
public class DataModelListViewModel : DataModelVisualizationViewModel
{
public DataModelListViewModel(PropertyInfo propertyInfo, DataModelPropertyAttribute propertyDescription, DataModelVisualizationViewModel parent)
{
PropertyInfo = propertyInfo;
Parent = parent;
PropertyDescription = propertyDescription;
Children = new BindableCollection<DataModelVisualizationViewModel>();
}
public BindableCollection<DataModelVisualizationViewModel> Children { get; set; }
public IList List { get; set; }
public string Count { get; set; }
public override void Update()
{
if (PropertyInfo != null && Parent?.Model != null && PropertyInfo.GetValue(Parent.Model) is IList listValue)
{
Model = new List<object>(listValue.Cast<object>());
List = (IList) Model;
}
var index = 0;
foreach (var item in List)
{
DataModelVisualizationViewModel child;
if (Children.Count < index )
{
child = CreateChild(item);
Children.Add(child);
}
else
{
// TODO: This wont fly when mixing DataModels and DataModelProperties
child = Children[index];
child.Model = item;
}
child.Update();
index++;
}
while (Children.Count > List.Count)
Children.RemoveAt(Children.Count - 1);
Count = $"{Children.Count} {(Children.Count == 1 ? "item" : "items")}";
}
}
}

View File

@ -1,39 +1,36 @@
using System; using System;
using System.Linq.Expressions;
using System.Reflection; using System.Reflection;
using Artemis.Core.Plugins.Abstract.DataModels.Attributes; using Artemis.Core.Plugins.Abstract.DataModels.Attributes;
namespace Artemis.UI.DataModelVisualization namespace Artemis.UI.DataModelVisualization
{ {
public class DataModelPropertyViewModel<T, TP> : DataModelPropertyViewModel public class DataModelPropertyViewModel : DataModelVisualizationViewModel
{ {
private readonly Func<T, TP> _expression; public DataModelPropertyViewModel(PropertyInfo propertyInfo, DataModelPropertyAttribute propertyDescription, DataModelVisualizationViewModel parent)
public DataModelPropertyViewModel(PropertyInfo propertyInfo, DataModelPropertyAttribute propertyDescription, DataModelViewModel parent)
{ {
PropertyInfo = propertyInfo; PropertyInfo = propertyInfo;
Parent = parent; Parent = parent;
PropertyDescription = propertyDescription; PropertyDescription = propertyDescription;
var instance = Expression.Parameter(typeof(T), "instance");
var body = Expression.Property(instance, propertyInfo);
_expression = Expression.Lambda<Func<T, TP>>(body, instance).Compile();
} }
public TP Value public bool IsListProperty { get; set; }
{ public string ListDescription { get; set; }
get => BaseValue is TP value ? value : default; public Type PropertyType { get; set; }
set => BaseValue = value;
}
public override void Update() public override void Update()
{ {
Value = _expression((T) Parent.Model); if (PropertyInfo != null && Parent?.Model != null)
}
}
public abstract class DataModelPropertyViewModel : DataModelVisualizationViewModel
{ {
public object BaseValue { get; protected set; } IsListProperty = false;
Model = PropertyInfo.GetValue(Parent.Model);
PropertyType = PropertyInfo.PropertyType;
}
else if (Parent is DataModelListViewModel listViewModel)
{
IsListProperty = true;
ListDescription = $"List item [{listViewModel.List.IndexOf(Model)}]";
PropertyType = Model.GetType();
}
}
} }
} }

View File

@ -14,7 +14,7 @@ namespace Artemis.UI.DataModelVisualization
Children = new BindableCollection<DataModelVisualizationViewModel>(); Children = new BindableCollection<DataModelVisualizationViewModel>();
} }
public DataModelViewModel(PropertyInfo propertyInfo, object model, DataModelPropertyAttribute propertyDescription, DataModelViewModel parent) public DataModelViewModel(PropertyInfo propertyInfo, object model, DataModelPropertyAttribute propertyDescription, DataModelVisualizationViewModel parent)
{ {
PropertyInfo = propertyInfo; PropertyInfo = propertyInfo;
Model = model; Model = model;
@ -25,7 +25,6 @@ namespace Artemis.UI.DataModelVisualization
PopulateProperties(); PopulateProperties();
} }
public object Model { get; private set; }
public BindableCollection<DataModelVisualizationViewModel> Children { get; set; } public BindableCollection<DataModelVisualizationViewModel> Children { get; set; }
public void PopulateProperties() public void PopulateProperties()
@ -33,33 +32,9 @@ namespace Artemis.UI.DataModelVisualization
Children.Clear(); Children.Clear();
foreach (var propertyInfo in Model.GetType().GetProperties()) foreach (var propertyInfo in Model.GetType().GetProperties())
{ {
// Skip properties decorated with DataModelIgnore var child = CreateChild(propertyInfo);
if (Attribute.IsDefined(propertyInfo, typeof(DataModelIgnoreAttribute))) if (child != null)
continue; Children.Add(child);
var dataModelPropertyAttribute = (DataModelPropertyAttribute) Attribute.GetCustomAttribute(propertyInfo, typeof(DataModelPropertyAttribute));
// If no DataModelProperty attribute was provided, pull one out of our ass
if (dataModelPropertyAttribute == null)
dataModelPropertyAttribute = new DataModelPropertyAttribute {Name = propertyInfo.Name.Humanize()};
// For primitives, create a property view model, it may be null that is fine
if (propertyInfo.PropertyType.IsPrimitive || propertyInfo.PropertyType == typeof(string))
{
// This may be slower than avoiding generics and Activator.CreateInstance but it allows for expression trees inside the VM we're creating
// here, this means slow creation but fast updates after that
var viewModelType = typeof(DataModelPropertyViewModel<,>).MakeGenericType(Model.GetType(), propertyInfo.PropertyType);
var viewModel = (DataModelVisualizationViewModel) Activator.CreateInstance(viewModelType, propertyInfo, dataModelPropertyAttribute, this);
Children.Add(viewModel);
}
// For other value types create a child view model if the value type is not null
else if (propertyInfo.PropertyType.IsClass || propertyInfo.PropertyType.IsStruct())
{
var value = propertyInfo.GetValue(Model);
if (value == null)
continue;
Children.Add(new DataModelViewModel(propertyInfo, value, dataModelPropertyAttribute, this));
}
} }
} }

View File

@ -0,0 +1,63 @@
using System;
using System.Collections;
using System.Reflection;
using Artemis.Core.Extensions;
using Artemis.Core.Plugins.Abstract.DataModels.Attributes;
using Humanizer;
using Stylet;
namespace Artemis.UI.DataModelVisualization
{
public abstract class DataModelVisualizationViewModel : PropertyChangedBase
{
public PropertyInfo PropertyInfo { get; protected set; }
public DataModelPropertyAttribute PropertyDescription { get; protected set; }
public DataModelVisualizationViewModel Parent { get; protected set; }
public object Model { get; set; }
public abstract void Update();
protected DataModelVisualizationViewModel CreateChild(PropertyInfo propertyInfo)
{
// Skip properties decorated with DataModelIgnore
if (Attribute.IsDefined(propertyInfo, typeof(DataModelIgnoreAttribute)))
return null;
var dataModelPropertyAttribute = (DataModelPropertyAttribute) Attribute.GetCustomAttribute(propertyInfo, typeof(DataModelPropertyAttribute));
// If no DataModelProperty attribute was provided, pull one out of our ass
if (dataModelPropertyAttribute == null)
dataModelPropertyAttribute = new DataModelPropertyAttribute {Name = propertyInfo.Name.Humanize()};
// For primitives, create a property view model, it may be null that is fine
if (propertyInfo.PropertyType.IsPrimitive || propertyInfo.PropertyType == typeof(string))
return new DataModelPropertyViewModel(propertyInfo, dataModelPropertyAttribute, this);
if (typeof(IList).IsAssignableFrom(propertyInfo.PropertyType))
return new DataModelListViewModel(propertyInfo, dataModelPropertyAttribute, this);
// For other value types create a child view model if the value type is not null
if (propertyInfo.PropertyType.IsClass || propertyInfo.PropertyType.IsStruct())
{
var value = propertyInfo.GetValue(Model);
if (value == null)
return null;
return new DataModelViewModel(propertyInfo, value, dataModelPropertyAttribute, this);
}
return null;
}
protected DataModelVisualizationViewModel CreateChild(object value)
{
var dataModelPropertyAttribute = new DataModelPropertyAttribute {Name = "Unknown property"};
// For primitives, create a property view model, it may be null that is fine
if (value.GetType().IsPrimitive || value is string)
return new DataModelPropertyViewModel(null, dataModelPropertyAttribute, this) {Model = value};
// For other value types create a child view model if the value type is not null
if (value.GetType().IsClass || value.GetType().IsStruct())
return new DataModelViewModel(null, value, dataModelPropertyAttribute, this);
return null;
}
}
}

View File

@ -19,6 +19,23 @@
<HierarchicalDataTemplate DataType="{x:Type dataModel:DataModelViewModel}" ItemsSource="{Binding Children}"> <HierarchicalDataTemplate DataType="{x:Type dataModel:DataModelViewModel}" ItemsSource="{Binding Children}">
<TextBlock Text="{Binding PropertyDescription.Name}" ToolTip="{Binding PropertyDescription.Description}" /> <TextBlock Text="{Binding PropertyDescription.Name}" ToolTip="{Binding PropertyDescription.Description}" />
</HierarchicalDataTemplate> </HierarchicalDataTemplate>
<HierarchicalDataTemplate DataType="{x:Type dataModel:DataModelListViewModel}" ItemsSource="{Binding Children}">
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
<TextBlock Grid.Column="0" Margin="0 0 5 0" FontWeight="Bold">
[List]
</TextBlock>
<TextBlock Grid.Column="1" Text="{Binding PropertyDescription.Name}" ToolTip="{Binding PropertyDescription.Description}" />
<TextBlock Grid.Column="2"
Text="{Binding Count, Mode=OneWay}"
FontFamily="Consolas"
HorizontalAlignment="Right"/>
</Grid>
</HierarchicalDataTemplate>
<HierarchicalDataTemplate DataType="{x:Type dataModel:DataModelPropertyViewModel}"> <HierarchicalDataTemplate DataType="{x:Type dataModel:DataModelPropertyViewModel}">
<Grid> <Grid>
<Grid.ColumnDefinitions> <Grid.ColumnDefinitions>
@ -27,20 +44,27 @@
<ColumnDefinition Width="*" /> <ColumnDefinition Width="*" />
</Grid.ColumnDefinitions> </Grid.ColumnDefinitions>
<TextBlock Grid.Column="0" Margin="0 0 5 0" FontWeight="Bold"> <TextBlock Grid.Column="0" Margin="0 0 5 0" FontWeight="Bold">
<Run>[</Run><Run Text="{Binding PropertyInfo.PropertyType.Name, Mode=OneWay}" /><Run>]</Run> <Run>[</Run><Run Text="{Binding PropertyType.Name, Mode=OneWay}" /><Run>]</Run>
</TextBlock> </TextBlock>
<TextBlock Grid.Column="1" Text="{Binding PropertyDescription.Name}" ToolTip="{Binding PropertyDescription.Description}" /> <TextBlock Grid.Column="1"
Text="{Binding PropertyDescription.Name}"
ToolTip="{Binding PropertyDescription.Description}"
Visibility="{Binding IsListProperty, Converter={x:Static s:BoolToVisibilityConverter.InverseInstance}, Mode=OneWay}"/>
<TextBlock Grid.Column="1"
Text="{Binding ListDescription}"
ToolTip="{Binding PropertyDescription.Description}"
Visibility="{Binding IsListProperty, Converter={x:Static s:BoolToVisibilityConverter.Instance}, Mode=OneWay}"/>
<TextBlock Grid.Column="2" <TextBlock Grid.Column="2"
Text="{Binding Value, Mode=OneWay}" Text="{Binding Model, Mode=OneWay}"
FontFamily="Consolas" FontFamily="Consolas"
HorizontalAlignment="Right" HorizontalAlignment="Right"
Visibility="{Binding Value, Converter={StaticResource NullToVisibilityConverter}}"/> Visibility="{Binding Model, Converter={StaticResource NullToVisibilityConverter}}"/>
<TextBlock Grid.Column="2" <TextBlock Grid.Column="2"
Text="null" Text="null"
FontFamily="Consolas" FontFamily="Consolas"
HorizontalAlignment="Right" HorizontalAlignment="Right"
Foreground="{DynamicResource MaterialDesignCheckBoxDisabled}" Foreground="{DynamicResource MaterialDesignCheckBoxDisabled}"
Visibility="{Binding Value, Converter={StaticResource NullToVisibilityConverter}, ConverterParameter=Inverted}"/> Visibility="{Binding Model, Converter={StaticResource NullToVisibilityConverter}, ConverterParameter=Inverted}"/>
</Grid> </Grid>
</HierarchicalDataTemplate> </HierarchicalDataTemplate>
</TreeView.Resources> </TreeView.Resources>

View File

@ -35,6 +35,8 @@ namespace Artemis.UI.Screens.Splash
_pluginService.CopyingBuildInPlugins += OnPluginServiceOnCopyingBuildInPlugins; _pluginService.CopyingBuildInPlugins += OnPluginServiceOnCopyingBuildInPlugins;
_pluginService.PluginLoading += OnPluginServiceOnPluginLoading; _pluginService.PluginLoading += OnPluginServiceOnPluginLoading;
_pluginService.PluginLoaded += OnPluginServiceOnPluginLoaded; _pluginService.PluginLoaded += OnPluginServiceOnPluginLoaded;
_pluginService.PluginEnabling += PluginServiceOnPluginEnabling;
_pluginService.PluginEnabled += PluginServiceOnPluginEnabled;
base.OnInitialActivate(); base.OnInitialActivate();
} }
@ -57,6 +59,16 @@ namespace Artemis.UI.Screens.Splash
Status = "Loading plugin: " + args.PluginInfo.Name; Status = "Loading plugin: " + args.PluginInfo.Name;
} }
private void PluginServiceOnPluginEnabled(object sender, PluginEventArgs args)
{
Status = "Initializing UI";
}
private void PluginServiceOnPluginEnabling(object sender, PluginEventArgs args)
{
Status = "Enabling plugin: " + args.PluginInfo.Name;
}
private void OnPluginServiceOnCopyingBuildInPlugins(object sender, EventArgs args) private void OnPluginServiceOnCopyingBuildInPlugins(object sender, EventArgs args)
{ {
Status = "Updating built-in plugins"; Status = "Updating built-in plugins";

View File

@ -1,4 +1,5 @@
using Artemis.Core.Plugins.Abstract.DataModels; using System.Collections.Generic;
using Artemis.Core.Plugins.Abstract.DataModels;
using Artemis.Core.Plugins.Abstract.DataModels.Attributes; using Artemis.Core.Plugins.Abstract.DataModels.Attributes;
using SkiaSharp; using SkiaSharp;
@ -9,6 +10,8 @@ namespace Artemis.Plugins.Modules.General
public GeneralDataModel() public GeneralDataModel()
{ {
PlayerInfo = new PlayerInfo(); PlayerInfo = new PlayerInfo();
IntsList = new List<int>();
PlayerInfosList = new List<PlayerInfo>();
} }
[DataModelProperty(Name = "A test string", Description = "This is a test string that's not of any use outside testing!")] [DataModelProperty(Name = "A test string", Description = "This is a test string that's not of any use outside testing!")]
@ -21,6 +24,9 @@ namespace Artemis.Plugins.Modules.General
public PlayerInfo PlayerInfo { get; set; } public PlayerInfo PlayerInfo { get; set; }
public double UpdatesDividedByFour { get; set; } public double UpdatesDividedByFour { get; set; }
public List<int> IntsList { get; set; }
public List<PlayerInfo> PlayerInfosList { get; set; }
} }
public class PlayerInfo : DataModel public class PlayerInfo : DataModel

View File

@ -1,7 +1,6 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using Artemis.Core.Plugins.Abstract; using Artemis.Core.Plugins.Abstract;
using Artemis.Core.Plugins.Abstract.DataModels;
using Artemis.Core.Plugins.Abstract.ViewModels; using Artemis.Core.Plugins.Abstract.ViewModels;
using Artemis.Core.Plugins.Models; using Artemis.Core.Plugins.Models;
using Artemis.Plugins.Modules.General.ViewModels; using Artemis.Plugins.Modules.General.ViewModels;
@ -12,7 +11,7 @@ namespace Artemis.Plugins.Modules.General
public class GeneralModule : ProfileModule<GeneralDataModel> public class GeneralModule : ProfileModule<GeneralDataModel>
{ {
private readonly PluginSettings _settings; private readonly PluginSettings _settings;
private Random _rand; private readonly Random _rand;
public GeneralModule(PluginSettings settings) public GeneralModule(PluginSettings settings)
{ {
@ -29,6 +28,9 @@ namespace Artemis.Plugins.Modules.General
{ {
DataModel.UpdatesDividedByFour += 0.25; DataModel.UpdatesDividedByFour += 0.25;
DataModel.PlayerInfo.Position = new SKPoint(_rand.Next(100), _rand.Next(100)); DataModel.PlayerInfo.Position = new SKPoint(_rand.Next(100), _rand.Next(100));
DataModel.IntsList[0] = _rand.Next();
DataModel.IntsList[2] = _rand.Next();
} }
public override void EnablePlugin() public override void EnablePlugin()
@ -37,6 +39,8 @@ namespace Artemis.Plugins.Modules.General
DisplayIcon = "AllInclusive"; DisplayIcon = "AllInclusive";
ExpandsDataModel = true; ExpandsDataModel = true;
DataModel.IntsList = new List<int> {_rand.Next(), _rand.Next(), _rand.Next()};
DataModel.PlayerInfosList = new List<PlayerInfo> {new PlayerInfo()};
var testSetting = _settings.GetSetting("TestSetting", DateTime.Now); var testSetting = _settings.GetSetting("TestSetting", DateTime.Now);
} }