mirror of
https://github.com/Artemis-RGB/Artemis
synced 2025-12-13 05:48:35 +00:00
Data model paths - Added support for lists
Data model paths - Added deferred compilation to accessors Data models - Fixed property name being empty sometimes Plugins - Fixed disabling plugins that failed to load
This commit is contained in:
parent
3be217a33b
commit
3fcfe4ceec
@ -1,6 +1,7 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections;
|
using System.Collections;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.ObjectModel;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Linq.Expressions;
|
using System.Linq.Expressions;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
@ -13,16 +14,17 @@ namespace Artemis.Core
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class DataModelPath
|
public class DataModelPath
|
||||||
{
|
{
|
||||||
|
private Expression<Func<object, object>> _accessorLambda;
|
||||||
private readonly LinkedList<DataModelPathSegment> _segments;
|
private readonly LinkedList<DataModelPathSegment> _segments;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates a new instance of the <see cref="DataModelPath" /> class
|
/// Creates a new instance of the <see cref="DataModelPath" /> class
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="dataModel">The data model at which this path starts</param>
|
/// <param name="target">The target at which this path starts</param>
|
||||||
/// <param name="path">A string representation of the <see cref="DataModelPath" /></param>
|
/// <param name="path">A string representation of the <see cref="DataModelPath" /></param>
|
||||||
public DataModelPath(object dataModel, string path)
|
public DataModelPath(object target, string path)
|
||||||
{
|
{
|
||||||
Target = dataModel ?? throw new ArgumentNullException(nameof(dataModel));
|
Target = target ?? throw new ArgumentNullException(nameof(target));
|
||||||
Path = path ?? throw new ArgumentNullException(nameof(path));
|
Path = path ?? throw new ArgumentNullException(nameof(path));
|
||||||
|
|
||||||
if (string.IsNullOrWhiteSpace(Path))
|
if (string.IsNullOrWhiteSpace(Path))
|
||||||
@ -53,34 +55,25 @@ namespace Artemis.Core
|
|||||||
public IReadOnlyCollection<DataModelPathSegment> Segments => _segments.ToList().AsReadOnly();
|
public IReadOnlyCollection<DataModelPathSegment> Segments => _segments.ToList().AsReadOnly();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets a boolean indicating whether this data model path can have an inner path because it points to a list
|
/// Gets a boolean indicating whether this data model path points to a list
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public bool CanHaveInnerPath => Segments.LastOrDefault()?.GetPropertyType()?.IsAssignableFrom(typeof(IList)) ?? false;
|
public bool PointsToList => Segments.LastOrDefault()?.GetPropertyType() != null &&
|
||||||
|
typeof(IList).IsAssignableFrom(Segments.LastOrDefault()?.GetPropertyType());
|
||||||
/// <summary>
|
|
||||||
/// Gets the inner path of this path, only available if this path points to a list
|
|
||||||
/// </summary>
|
|
||||||
public DataModelPath InnerPath { get; internal set; }
|
|
||||||
|
|
||||||
internal Func<object, object> Accessor { get; private set; }
|
internal Func<object, object> Accessor { get; private set; }
|
||||||
|
|
||||||
public void SetInnerPath(string path)
|
|
||||||
{
|
|
||||||
if (!CanHaveInnerPath)
|
|
||||||
{
|
|
||||||
var type = Segments.LastOrDefault()?.GetPropertyType();
|
|
||||||
throw new ArtemisCoreException($"Cannot set an inner path on a data model path if it does not point to a list (value is of type: {type?.Name})");
|
|
||||||
}
|
|
||||||
|
|
||||||
InnerPath = new DataModelPath(GetValue(), path);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the current value of the path
|
/// Gets the current value of the path
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public object GetValue()
|
public object GetValue()
|
||||||
{
|
{
|
||||||
return Accessor?.Invoke(Target);
|
if (_accessorLambda == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
// If the accessor has not yet been compiled do it now that it's first required
|
||||||
|
if (Accessor == null)
|
||||||
|
Accessor = _accessorLambda.Compile();
|
||||||
|
return Accessor(Target);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -89,8 +82,6 @@ namespace Artemis.Core
|
|||||||
/// <returns>If static, the property info. If dynamic, <c>null</c></returns>
|
/// <returns>If static, the property info. If dynamic, <c>null</c></returns>
|
||||||
public PropertyInfo GetPropertyInfo()
|
public PropertyInfo GetPropertyInfo()
|
||||||
{
|
{
|
||||||
if (InnerPath != null)
|
|
||||||
return InnerPath.GetPropertyInfo();
|
|
||||||
return Segments.LastOrDefault()?.GetPropertyInfo();
|
return Segments.LastOrDefault()?.GetPropertyInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -100,8 +91,6 @@ namespace Artemis.Core
|
|||||||
/// <returns>If possible, the property type</returns>
|
/// <returns>If possible, the property type</returns>
|
||||||
public Type GetPropertyType()
|
public Type GetPropertyType()
|
||||||
{
|
{
|
||||||
if (InnerPath != null)
|
|
||||||
return InnerPath.GetPropertyType();
|
|
||||||
return Segments.LastOrDefault()?.GetPropertyType();
|
return Segments.LastOrDefault()?.GetPropertyType();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,21 +100,20 @@ namespace Artemis.Core
|
|||||||
/// <returns>If found, the data model property description</returns>
|
/// <returns>If found, the data model property description</returns>
|
||||||
public DataModelPropertyAttribute GetPropertyDescription()
|
public DataModelPropertyAttribute GetPropertyDescription()
|
||||||
{
|
{
|
||||||
if (InnerPath != null)
|
|
||||||
return InnerPath.GetPropertyDescription();
|
|
||||||
return Segments.LastOrDefault()?.GetPropertyDescription();
|
return Segments.LastOrDefault()?.GetPropertyDescription();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
public override string ToString()
|
public override string ToString()
|
||||||
{
|
{
|
||||||
if (InnerPath != null)
|
|
||||||
return $"{Path} > {InnerPath}";
|
|
||||||
return Path;
|
return Path;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Initialize(string path)
|
private void Initialize(string path)
|
||||||
{
|
{
|
||||||
|
var startSegment = new DataModelPathSegment(this, "target", "target");
|
||||||
|
startSegment.Node = _segments.AddFirst(startSegment);
|
||||||
|
|
||||||
var segments = path.Split(".");
|
var segments = path.Split(".");
|
||||||
for (var index = 0; index < segments.Length; index++)
|
for (var index = 0; index < segments.Length; index++)
|
||||||
{
|
{
|
||||||
@ -147,7 +135,7 @@ namespace Artemis.Core
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Accessor = Expression.Lambda<Func<object, object>>(
|
_accessorLambda = Expression.Lambda<Func<object, object>>(
|
||||||
// Wrap with a null check
|
// Wrap with a null check
|
||||||
Expression.Condition(
|
Expression.Condition(
|
||||||
nullCondition,
|
nullCondition,
|
||||||
@ -155,7 +143,8 @@ namespace Artemis.Core
|
|||||||
Expression.Convert(Expression.Default(expression.Type), typeof(object))
|
Expression.Convert(Expression.Default(expression.Type), typeof(object))
|
||||||
),
|
),
|
||||||
parameter
|
parameter
|
||||||
).Compile();
|
);
|
||||||
|
Accessor = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
using System.Linq.Expressions;
|
using System.Linq.Expressions;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using Artemis.Core.DataModelExpansions;
|
using Artemis.Core.DataModelExpansions;
|
||||||
@ -12,11 +13,14 @@ namespace Artemis.Core
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class DataModelPathSegment
|
public class DataModelPathSegment
|
||||||
{
|
{
|
||||||
|
private Expression<Func<object, object>> _accessorLambda;
|
||||||
|
|
||||||
internal DataModelPathSegment(DataModelPath dataModelPath, string identifier, string path)
|
internal DataModelPathSegment(DataModelPath dataModelPath, string identifier, string path)
|
||||||
{
|
{
|
||||||
DataModelPath = dataModelPath;
|
DataModelPath = dataModelPath;
|
||||||
Identifier = identifier;
|
Identifier = identifier;
|
||||||
Path = path;
|
Path = path;
|
||||||
|
IsStartSegment = !DataModelPath.Segments.Any();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -34,6 +38,11 @@ namespace Artemis.Core
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public string Path { get; }
|
public string Path { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets a boolean indicating whether this is the first segment in the path
|
||||||
|
/// </summary>
|
||||||
|
public bool IsStartSegment { get; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the type of data model this segment of the path points to
|
/// Gets the type of data model this segment of the path points to
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -64,7 +73,13 @@ namespace Artemis.Core
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public object GetValue()
|
public object GetValue()
|
||||||
{
|
{
|
||||||
return Type == DataModelPathSegmentType.Invalid ? null : Accessor(DataModelPath.Target);
|
if (Type == DataModelPathSegmentType.Invalid || _accessorLambda == null)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
// If the accessor has not yet been compiled do it now that it's first required
|
||||||
|
if (Accessor == null)
|
||||||
|
Accessor = _accessorLambda.Compile();
|
||||||
|
return Accessor(DataModelPath.Target);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <inheritdoc />
|
/// <inheritdoc />
|
||||||
@ -82,12 +97,12 @@ namespace Artemis.Core
|
|||||||
// Dynamic types have no property and therefore no property info
|
// Dynamic types have no property and therefore no property info
|
||||||
if (Type == DataModelPathSegmentType.Dynamic)
|
if (Type == DataModelPathSegmentType.Dynamic)
|
||||||
return null;
|
return null;
|
||||||
|
// The start segment has none either because it is the datamodel
|
||||||
|
if (IsStartSegment)
|
||||||
|
return null;
|
||||||
|
|
||||||
// If this is the first segment in a path, the property is located on the data model
|
|
||||||
if (Previous == null)
|
|
||||||
return DataModelPath.Target.GetType().GetProperty(Identifier);
|
|
||||||
// If this is not the first segment in a path, the property is located on the previous segment
|
// If this is not the first segment in a path, the property is located on the previous segment
|
||||||
return Previous.GetValue()?.GetType().GetProperty(Identifier);
|
return Previous.GetPropertyType()?.GetProperty(Identifier);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -99,14 +114,24 @@ namespace Artemis.Core
|
|||||||
// Dynamic types have a data model description
|
// Dynamic types have a data model description
|
||||||
if (Type == DataModelPathSegmentType.Dynamic)
|
if (Type == DataModelPathSegmentType.Dynamic)
|
||||||
return ((DataModel) GetValue())?.DataModelDescription;
|
return ((DataModel) GetValue())?.DataModelDescription;
|
||||||
|
if (IsStartSegment && DataModelPath.Target is DataModel targetDataModel)
|
||||||
|
return targetDataModel.DataModelDescription;
|
||||||
|
if (IsStartSegment)
|
||||||
|
return null;
|
||||||
|
|
||||||
var propertyInfo = GetPropertyInfo();
|
var propertyInfo = GetPropertyInfo();
|
||||||
if (propertyInfo == null)
|
if (propertyInfo == null)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
// Static types may have one as an attribute
|
// Static types may have one as an attribute
|
||||||
return (DataModelPropertyAttribute) Attribute.GetCustomAttribute(propertyInfo, typeof(DataModelPropertyAttribute)) ??
|
var attribute = (DataModelPropertyAttribute) Attribute.GetCustomAttribute(propertyInfo, typeof(DataModelPropertyAttribute));
|
||||||
new DataModelPropertyAttribute {Name = propertyInfo.Name.Humanize(), ResetsDepth = false};
|
if (attribute != null)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(attribute.Name))
|
||||||
|
attribute.Name = propertyInfo.Name.Humanize();
|
||||||
|
return attribute;
|
||||||
|
}
|
||||||
|
return new DataModelPropertyAttribute {Name = propertyInfo.Name.Humanize(), ResetsDepth = false};
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -115,6 +140,10 @@ namespace Artemis.Core
|
|||||||
/// <returns>If possible, the property type</returns>
|
/// <returns>If possible, the property type</returns>
|
||||||
public Type GetPropertyType()
|
public Type GetPropertyType()
|
||||||
{
|
{
|
||||||
|
// The start segment type is always the target type
|
||||||
|
if (IsStartSegment)
|
||||||
|
return DataModelPath.Target.GetType();
|
||||||
|
|
||||||
// Prefer basing the type on the property info
|
// Prefer basing the type on the property info
|
||||||
var propertyInfo = GetPropertyInfo();
|
var propertyInfo = GetPropertyInfo();
|
||||||
var type = propertyInfo?.PropertyType;
|
var type = propertyInfo?.PropertyType;
|
||||||
@ -131,27 +160,35 @@ namespace Artemis.Core
|
|||||||
|
|
||||||
internal Expression Initialize(ParameterExpression parameter, Expression expression, Expression nullCondition)
|
internal Expression Initialize(ParameterExpression parameter, Expression expression, Expression nullCondition)
|
||||||
{
|
{
|
||||||
var previousValue = Previous != null ? Previous.GetValue() : DataModelPath.Target;
|
if (IsStartSegment)
|
||||||
if (previousValue == null)
|
|
||||||
{
|
{
|
||||||
Type = DataModelPathSegmentType.Invalid;
|
Type = DataModelPathSegmentType.Static;
|
||||||
return null;
|
return CreateExpression(parameter, expression, nullCondition);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Determine this segment's type by looking for a dynamic data model with the identifier
|
var previousType = Previous.GetPropertyType();
|
||||||
if (previousValue is DataModel dataModel)
|
if (previousType == null)
|
||||||
{
|
{
|
||||||
var hasDynamicDataModel = dataModel.DynamicDataModels.TryGetValue(Identifier, out var dynamicDataModel);
|
Type = DataModelPathSegmentType.Invalid;
|
||||||
|
return CreateExpression(parameter, expression, nullCondition);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prefer static since that's faster
|
||||||
|
DetermineStaticType(previousType);
|
||||||
|
|
||||||
|
// If no static type could be found, check if this is a data model and if so, look for a dynamic type
|
||||||
|
if (Type == DataModelPathSegmentType.Invalid && typeof(DataModel).IsAssignableFrom(previousType))
|
||||||
|
{
|
||||||
|
var dataModel = (DataModel) Previous.GetValue();
|
||||||
|
// Cannot determine a dynamic type on a null data model, leave the segment invalid
|
||||||
|
if (dataModel == null)
|
||||||
|
return CreateExpression(parameter, expression, nullCondition);
|
||||||
|
|
||||||
// If a dynamic data model is found the use that
|
// If a dynamic data model is found the use that
|
||||||
|
var hasDynamicDataModel = dataModel.DynamicDataModels.TryGetValue(Identifier, out var dynamicDataModel);
|
||||||
if (hasDynamicDataModel)
|
if (hasDynamicDataModel)
|
||||||
DetermineDynamicType(dynamicDataModel);
|
DetermineDynamicType(dynamicDataModel);
|
||||||
// Otherwise look for a static type
|
|
||||||
else
|
|
||||||
DetermineStaticType(previousValue);
|
|
||||||
}
|
}
|
||||||
// Only data models can have dynamic types so if it is something else, its always going to be static
|
|
||||||
else
|
|
||||||
DetermineStaticType(previousValue);
|
|
||||||
|
|
||||||
return CreateExpression(parameter, expression, nullCondition);
|
return CreateExpression(parameter, expression, nullCondition);
|
||||||
}
|
}
|
||||||
@ -160,13 +197,17 @@ namespace Artemis.Core
|
|||||||
{
|
{
|
||||||
if (Type == DataModelPathSegmentType.Invalid)
|
if (Type == DataModelPathSegmentType.Invalid)
|
||||||
{
|
{
|
||||||
|
_accessorLambda = null;
|
||||||
Accessor = null;
|
Accessor = null;
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Expression accessorExpression;
|
Expression accessorExpression;
|
||||||
|
// A start segment just accesses the target
|
||||||
|
if (IsStartSegment)
|
||||||
|
accessorExpression = expression;
|
||||||
// A static segment just needs to access the property or filed
|
// A static segment just needs to access the property or filed
|
||||||
if (Type == DataModelPathSegmentType.Static)
|
else if (Type == DataModelPathSegmentType.Static)
|
||||||
accessorExpression = Expression.PropertyOrField(expression, Identifier);
|
accessorExpression = Expression.PropertyOrField(expression, Identifier);
|
||||||
// A dynamic segment calls the generic method DataModel.DynamicChild<T> and provides the identifier as an argument
|
// A dynamic segment calls the generic method DataModel.DynamicChild<T> and provides the identifier as an argument
|
||||||
else
|
else
|
||||||
@ -179,7 +220,7 @@ namespace Artemis.Core
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
Accessor = Expression.Lambda<Func<object, object>>(
|
_accessorLambda = Expression.Lambda<Func<object, object>>(
|
||||||
// Wrap with a null check
|
// Wrap with a null check
|
||||||
Expression.Condition(
|
Expression.Condition(
|
||||||
nullCondition,
|
nullCondition,
|
||||||
@ -187,8 +228,8 @@ namespace Artemis.Core
|
|||||||
Expression.Convert(Expression.Default(accessorExpression.Type), typeof(object))
|
Expression.Convert(Expression.Default(accessorExpression.Type), typeof(object))
|
||||||
),
|
),
|
||||||
parameter
|
parameter
|
||||||
).Compile();
|
);
|
||||||
|
Accessor = null;
|
||||||
return accessorExpression;
|
return accessorExpression;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -198,9 +239,8 @@ namespace Artemis.Core
|
|||||||
DynamicDataModelType = dynamicDataModel.GetType();
|
DynamicDataModelType = dynamicDataModel.GetType();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void DetermineStaticType(object previous)
|
private void DetermineStaticType(Type previousType)
|
||||||
{
|
{
|
||||||
var previousType = previous.GetType();
|
|
||||||
var property = previousType.GetProperty(Identifier, BindingFlags.Public | BindingFlags.Instance);
|
var property = previousType.GetProperty(Identifier, BindingFlags.Public | BindingFlags.Instance);
|
||||||
Type = property == null ? DataModelPathSegmentType.Invalid : DataModelPathSegmentType.Static;
|
Type = property == null ? DataModelPathSegmentType.Invalid : DataModelPathSegmentType.Static;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -64,7 +64,8 @@ namespace Artemis.Core.DataModelExpansions
|
|||||||
throw new ArgumentNullException(nameof(dynamicDataModel));
|
throw new ArgumentNullException(nameof(dynamicDataModel));
|
||||||
if (key == null)
|
if (key == null)
|
||||||
throw new ArgumentNullException(nameof(key));
|
throw new ArgumentNullException(nameof(key));
|
||||||
|
if (key.Contains('.'))
|
||||||
|
throw new ArtemisCoreException("The provided key contains an illegal character (.)");
|
||||||
if (_dynamicDataModels.ContainsKey(key))
|
if (_dynamicDataModels.ContainsKey(key))
|
||||||
{
|
{
|
||||||
throw new ArtemisCoreException($"Cannot add a dynamic data model with key '{key}' " +
|
throw new ArtemisCoreException($"Cannot add a dynamic data model with key '{key}' " +
|
||||||
@ -87,7 +88,7 @@ namespace Artemis.Core.DataModelExpansions
|
|||||||
dynamicDataModel.PluginInfo = PluginInfo;
|
dynamicDataModel.PluginInfo = PluginInfo;
|
||||||
dynamicDataModel.DataModelDescription = new DataModelPropertyAttribute
|
dynamicDataModel.DataModelDescription = new DataModelPropertyAttribute
|
||||||
{
|
{
|
||||||
Name = name ?? key.Humanize(),
|
Name = string.IsNullOrWhiteSpace(name) ? key.Humanize() : name,
|
||||||
Description = description
|
Description = description
|
||||||
};
|
};
|
||||||
_dynamicDataModels.Add(key, dynamicDataModel);
|
_dynamicDataModels.Add(key, dynamicDataModel);
|
||||||
|
|||||||
@ -105,6 +105,12 @@ namespace Artemis.Core
|
|||||||
InternalDisablePlugin();
|
InternalDisablePlugin();
|
||||||
OnPluginDisabled();
|
OnPluginDisabled();
|
||||||
}
|
}
|
||||||
|
// A failed load is still enabled in plugin info (to avoid disabling it permanently after a fail)
|
||||||
|
// update even that when manually disabling
|
||||||
|
else if (!enable && !Enabled)
|
||||||
|
{
|
||||||
|
PluginInfo.Enabled = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal virtual void InternalEnablePlugin()
|
internal virtual void InternalEnablePlugin()
|
||||||
|
|||||||
@ -1,5 +1,4 @@
|
|||||||
using System;
|
using System;
|
||||||
using Artemis.Core;
|
|
||||||
using Artemis.Core.DataModelExpansions;
|
using Artemis.Core.DataModelExpansions;
|
||||||
using Artemis.UI.Shared.Services;
|
using Artemis.UI.Shared.Services;
|
||||||
|
|
||||||
@ -11,8 +10,11 @@ namespace Artemis.UI.Shared
|
|||||||
private int _index;
|
private int _index;
|
||||||
private Type _listType;
|
private Type _listType;
|
||||||
|
|
||||||
public DataModelListPropertiesViewModel(DataModel dataModel, DataModelVisualizationViewModel parent, DataModelPath dataModelPath) : base(dataModel, parent, dataModelPath)
|
public DataModelListPropertiesViewModel(DataModel dataModel, object listItem) : base(null, null, null)
|
||||||
{
|
{
|
||||||
|
DataModel = dataModel;
|
||||||
|
ListType = listItem.GetType();
|
||||||
|
DisplayValue = listItem;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Index
|
public int Index
|
||||||
@ -42,7 +44,7 @@ namespace Artemis.UI.Shared
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
ListType = DisplayValue.GetType();
|
ListType = DisplayValue.GetType();
|
||||||
PopulateProperties(dataModelUIService);
|
PopulateProperties(dataModelUIService, DisplayValue);
|
||||||
foreach (var dataModelVisualizationViewModel in Children)
|
foreach (var dataModelVisualizationViewModel in Children)
|
||||||
dataModelVisualizationViewModel.Update(dataModelUIService);
|
dataModelVisualizationViewModel.Update(dataModelUIService);
|
||||||
}
|
}
|
||||||
@ -51,5 +53,11 @@ namespace Artemis.UI.Shared
|
|||||||
{
|
{
|
||||||
return DisplayValue;
|
return DisplayValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"[List item {Index}] {DisplayPath ?? Path}";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,5 +1,4 @@
|
|||||||
using System;
|
using System;
|
||||||
using Artemis.Core;
|
|
||||||
using Artemis.Core.DataModelExpansions;
|
using Artemis.Core.DataModelExpansions;
|
||||||
using Artemis.UI.Shared.Services;
|
using Artemis.UI.Shared.Services;
|
||||||
|
|
||||||
@ -10,8 +9,19 @@ namespace Artemis.UI.Shared
|
|||||||
private int _index;
|
private int _index;
|
||||||
private Type _listType;
|
private Type _listType;
|
||||||
|
|
||||||
public DataModelListPropertyViewModel(DataModel dataModel, DataModelVisualizationViewModel parent, DataModelPath dataModelPath) : base(dataModel, parent, dataModelPath)
|
public DataModelListPropertyViewModel(DataModel dataModel, object listItem, DataModelDisplayViewModel displayViewModel) : base(null, null, null)
|
||||||
{
|
{
|
||||||
|
DataModel = dataModel;
|
||||||
|
ListType = listItem.GetType();
|
||||||
|
DisplayValue = listItem;
|
||||||
|
DisplayViewModel = displayViewModel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DataModelListPropertyViewModel(DataModel dataModel, object listItem) : base(null, null, null)
|
||||||
|
{
|
||||||
|
DataModel = dataModel;
|
||||||
|
ListType = listItem.GetType();
|
||||||
|
DisplayValue = listItem;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Index
|
public int Index
|
||||||
@ -43,5 +53,11 @@ namespace Artemis.UI.Shared
|
|||||||
ListType = DisplayValue.GetType();
|
ListType = DisplayValue.GetType();
|
||||||
UpdateDisplayParameters();
|
UpdateDisplayParameters();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"[List item {Index}] {DisplayPath ?? Path} - {DisplayValue}";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -105,20 +105,25 @@ namespace Artemis.UI.Shared
|
|||||||
protected DataModelVisualizationViewModel CreateListChild(IDataModelUIService dataModelUIService, object listItem)
|
protected DataModelVisualizationViewModel CreateListChild(IDataModelUIService dataModelUIService, object listItem)
|
||||||
{
|
{
|
||||||
var listType = listItem.GetType();
|
var listType = listItem.GetType();
|
||||||
var path = new DataModelPath(listItem, "");
|
|
||||||
|
|
||||||
// If a display VM was found, prefer to use that in any case
|
// If a display VM was found, prefer to use that in any case
|
||||||
var typeViewModel = dataModelUIService.GetDataModelDisplayViewModel(listType);
|
var typeViewModel = dataModelUIService.GetDataModelDisplayViewModel(listType);
|
||||||
if (typeViewModel != null)
|
if (typeViewModel != null)
|
||||||
return new DataModelListPropertyViewModel(DataModel, this, path) {DisplayViewModel = typeViewModel};
|
return new DataModelListPropertyViewModel(DataModel, listItem, typeViewModel);
|
||||||
// For primitives, create a property view model, it may be null that is fine
|
// For primitives, create a property view model, it may be null that is fine
|
||||||
if (listType.IsPrimitive || listType.IsEnum || listType == typeof(string))
|
if (listType.IsPrimitive || listType.IsEnum || listType == typeof(string))
|
||||||
return new DataModelListPropertyViewModel(DataModel, this, path) {ListType = listType};
|
return new DataModelListPropertyViewModel(DataModel, listItem);
|
||||||
// For other value types create a child view model
|
// For other value types create a child view model
|
||||||
if (listType.IsClass || listType.IsStruct())
|
if (listType.IsClass || listType.IsStruct())
|
||||||
return new DataModelListPropertiesViewModel(DataModel, this, path) {ListType = listType};
|
return new DataModelListPropertiesViewModel(DataModel, listItem);
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"[List] {DisplayPath ?? Path} - {List.Count} item(s)";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -13,7 +13,7 @@ namespace Artemis.UI.Shared
|
|||||||
public override void Update(IDataModelUIService dataModelUIService)
|
public override void Update(IDataModelUIService dataModelUIService)
|
||||||
{
|
{
|
||||||
// Always populate properties
|
// Always populate properties
|
||||||
PopulateProperties(dataModelUIService);
|
PopulateProperties(dataModelUIService, null);
|
||||||
|
|
||||||
// Only update children if the parent is expanded
|
// Only update children if the parent is expanded
|
||||||
if (Parent != null && !Parent.IsVisualizationExpanded && !Parent.IsRootViewModel)
|
if (Parent != null && !Parent.IsVisualizationExpanded && !Parent.IsRootViewModel)
|
||||||
@ -28,9 +28,15 @@ namespace Artemis.UI.Shared
|
|||||||
return Parent.IsRootViewModel ? DataModel : base.GetCurrentValue();
|
return Parent.IsRootViewModel ? DataModel : base.GetCurrentValue();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override int GetChildDepth()
|
internal override int GetChildDepth()
|
||||||
{
|
{
|
||||||
return PropertyDescription != null && !PropertyDescription.ResetsDepth ? Depth + 1 : 1;
|
return PropertyDescription != null && !PropertyDescription.ResetsDepth ? Depth + 1 : 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return DisplayPath ?? Path;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -50,5 +50,11 @@ namespace Artemis.UI.Shared
|
|||||||
{
|
{
|
||||||
DisplayViewModel?.UpdateValue(DisplayValue);
|
DisplayViewModel?.UpdateValue(DisplayValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"[{DisplayValueType.Name}] {DisplayPath ?? Path} - {DisplayValue}";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections;
|
using System.Collections;
|
||||||
|
using System.Collections.ObjectModel;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using Artemis.Core;
|
using Artemis.Core;
|
||||||
@ -136,21 +137,6 @@ namespace Artemis.UI.Shared
|
|||||||
IsMatchingFilteredTypes = filteredTypes.Any(t => t == type || t == typeof(Enum) && type.IsEnum);
|
IsMatchingFilteredTypes = filteredTypes.Any(t => t == type || t == typeof(Enum) && type.IsEnum);
|
||||||
}
|
}
|
||||||
|
|
||||||
public DataModelVisualizationViewModel GetChildForCondition(DataModelConditionPredicate predicate, DataModelConditionSide side)
|
|
||||||
{
|
|
||||||
if (side == DataModelConditionSide.Left)
|
|
||||||
{
|
|
||||||
if (predicate.LeftDataModel == null || predicate.LeftPropertyPath == null)
|
|
||||||
return null;
|
|
||||||
|
|
||||||
return GetChildByPath(predicate.LeftDataModel.PluginInfo.Guid, predicate.LeftPropertyPath);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (predicate.RightDataModel == null || predicate.RightPropertyPath == null)
|
|
||||||
return null;
|
|
||||||
return GetChildByPath(predicate.RightDataModel.PluginInfo.Guid, predicate.RightPropertyPath);
|
|
||||||
}
|
|
||||||
|
|
||||||
public DataModelVisualizationViewModel GetChildByPath(Guid dataModelGuid, string propertyPath)
|
public DataModelVisualizationViewModel GetChildByPath(Guid dataModelGuid, string propertyPath)
|
||||||
{
|
{
|
||||||
if (DataModel.PluginInfo.Guid != dataModelGuid)
|
if (DataModel.PluginInfo.Guid != dataModelGuid)
|
||||||
@ -183,31 +169,44 @@ namespace Artemis.UI.Shared
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected virtual int GetChildDepth()
|
internal virtual int GetChildDepth()
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void PopulateProperties(IDataModelUIService dataModelUIService)
|
internal void PopulateProperties(IDataModelUIService dataModelUIService, object overrideValue)
|
||||||
{
|
{
|
||||||
if (IsRootViewModel)
|
if (IsRootViewModel && overrideValue == null)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
Type modelType;
|
||||||
|
if (overrideValue != null)
|
||||||
|
modelType = overrideValue.GetType();
|
||||||
|
else if (Parent.IsRootViewModel)
|
||||||
|
modelType = DataModel.GetType();
|
||||||
|
else
|
||||||
|
modelType = DataModelPath.GetPropertyType();
|
||||||
|
|
||||||
// Add missing static children
|
// Add missing static children
|
||||||
var modelType = Parent.IsRootViewModel ? DataModel.GetType() : DataModelPath.GetPropertyType();
|
|
||||||
foreach (var propertyInfo in modelType.GetProperties(BindingFlags.Public | BindingFlags.Instance))
|
foreach (var propertyInfo in modelType.GetProperties(BindingFlags.Public | BindingFlags.Instance))
|
||||||
{
|
{
|
||||||
var childPath = Path != null ? $"{Path}.{propertyInfo.Name}" : propertyInfo.Name;
|
var childPath = Path != null ? $"{Path}.{propertyInfo.Name}" : propertyInfo.Name;
|
||||||
if (Children.Any(c => c.Path != null && c.Path.Equals(childPath)))
|
if (Children.Any(c => c.Path != null && c.Path.Equals(childPath)))
|
||||||
continue;
|
continue;
|
||||||
|
if (propertyInfo.GetCustomAttribute<DataModelIgnoreAttribute>() != null)
|
||||||
|
continue;
|
||||||
|
|
||||||
var child = CreateChild(dataModelUIService, childPath, GetChildDepth());
|
var child = CreateChild(dataModelUIService, childPath, GetChildDepth(), overrideValue);
|
||||||
if (child != null)
|
if (child != null)
|
||||||
Children.Add(child);
|
Children.Add(child);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove static children that should be hidden
|
// Remove static children that should be hidden
|
||||||
var hiddenProperties = DataModel.GetHiddenProperties();
|
ReadOnlyCollection<PropertyInfo> hiddenProperties;
|
||||||
|
if (overrideValue != null && overrideValue is DataModel overrideValueDataModel)
|
||||||
|
hiddenProperties = overrideValueDataModel.GetHiddenProperties();
|
||||||
|
else
|
||||||
|
hiddenProperties = DataModel.GetHiddenProperties();
|
||||||
foreach (var hiddenProperty in hiddenProperties)
|
foreach (var hiddenProperty in hiddenProperties)
|
||||||
{
|
{
|
||||||
var childPath = Path != null ? $"{Path}.{hiddenProperty.Name}" : hiddenProperty.Name;
|
var childPath = Path != null ? $"{Path}.{hiddenProperty.Name}" : hiddenProperty.Name;
|
||||||
@ -217,7 +216,11 @@ namespace Artemis.UI.Shared
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Add missing dynamic children
|
// Add missing dynamic children
|
||||||
var value = Parent.IsRootViewModel ? DataModel : DataModelPath.GetValue();
|
object value;
|
||||||
|
if (overrideValue != null)
|
||||||
|
value = overrideValue;
|
||||||
|
else
|
||||||
|
value = Parent.IsRootViewModel ? DataModel : DataModelPath.GetValue();
|
||||||
if (value is DataModel dataModel)
|
if (value is DataModel dataModel)
|
||||||
{
|
{
|
||||||
foreach (var kvp in dataModel.DynamicDataModels)
|
foreach (var kvp in dataModel.DynamicDataModels)
|
||||||
@ -226,7 +229,7 @@ namespace Artemis.UI.Shared
|
|||||||
if (Children.Any(c => c.Path != null && c.Path.Equals(childPath)))
|
if (Children.Any(c => c.Path != null && c.Path.Equals(childPath)))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
var child = CreateChild(dataModelUIService, childPath, GetChildDepth());
|
var child = CreateChild(dataModelUIService, childPath, GetChildDepth(), overrideValue);
|
||||||
if (child != null)
|
if (child != null)
|
||||||
Children.Add(child);
|
Children.Add(child);
|
||||||
}
|
}
|
||||||
@ -238,12 +241,12 @@ namespace Artemis.UI.Shared
|
|||||||
Children.RemoveRange(toRemoveDynamic);
|
Children.RemoveRange(toRemoveDynamic);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected DataModelVisualizationViewModel CreateChild(IDataModelUIService dataModelUIService, string path, int depth)
|
private DataModelVisualizationViewModel CreateChild(IDataModelUIService dataModelUIService, string path, int depth, object overrideValue)
|
||||||
{
|
{
|
||||||
if (depth > MaxDepth)
|
if (depth > MaxDepth)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
var dataModelPath = new DataModelPath(DataModel, path);
|
var dataModelPath = new DataModelPath(overrideValue ?? DataModel, path);
|
||||||
if (!dataModelPath.IsValid)
|
if (!dataModelPath.IsValid)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user