Xamarin Public Jenkins (auto-signing) e79aa3c0ed Imported Upstream version 4.6.0.125
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
2016-08-03 10:59:49 +00:00

1910 lines
75 KiB
C#

namespace System.Workflow.ComponentModel
{
#region Using directives
using System;
using System.IO;
using System.Xml;
using System.Text;
using System.CodeDom;
using System.Reflection;
using System.Xml.XPath;
using System.Collections;
using System.Xml.Serialization;
using System.Collections.Generic;
using System.ComponentModel;
using System.Collections.Specialized;
using System.ComponentModel.Design;
using System.ComponentModel.Design.Serialization;
using System.Diagnostics;
using System.Globalization;
using System.Drawing.Design;
using System.Workflow.ComponentModel.Compiler;
using System.Workflow.ComponentModel.Serialization;
using System.Workflow.ComponentModel.Design;
using System.Configuration;
#endregion
#region Bind
[Browsable(false)]
internal abstract class BindBase
{
[NonSerialized]
protected bool designMode = true;
[NonSerialized]
private object syncRoot = new object();
public abstract object GetRuntimeValue(Activity activity);
public abstract object GetRuntimeValue(Activity activity, Type targetType);
public abstract void SetRuntimeValue(Activity activity, object value);
protected virtual void OnRuntimeInitialized(Activity activity)
{
}
}
#endregion
#region Redundant Binds
#region MemberBind
//
internal abstract class MemberBind : BindBase
{
private string name = string.Empty;
protected MemberBind()
{
}
protected MemberBind(string name)
{
this.name = name;
}
[DefaultValue("")]
public string Name
{
get
{
return this.name;
}
}
internal static object GetValue(MemberInfo memberInfo, object dataContext, string path)
{
if (memberInfo == null)
throw new ArgumentNullException("memberInfo");
if (dataContext == null)
throw new ArgumentNullException("dataContext");
if (path == null)
path = string.Empty;
if (string.IsNullOrEmpty(path))
return null;
object targetObject = dataContext;
System.Type memberType = dataContext.GetType();
PathWalker pathWalker = new PathWalker();
pathWalker.MemberFound += delegate(object sender, PathMemberInfoEventArgs eventArgs)
{
if (targetObject == null)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
switch (eventArgs.MemberKind)
{
case PathMemberKind.Field:
memberType = (eventArgs.MemberInfo as FieldInfo).FieldType;
targetObject = (eventArgs.MemberInfo as FieldInfo).GetValue(targetObject);
break;
case PathMemberKind.Event:
EventInfo evt = eventArgs.MemberInfo as EventInfo;
memberType = evt.EventHandlerType;
// GetValue() returns the actual value of the property. We need the Bind object here.
// Find out if there is a matching dependency property and get the value throw the DP.
DependencyObject dependencyObject = targetObject as DependencyObject;
DependencyProperty dependencyProperty = DependencyProperty.FromName(evt.Name, dependencyObject.GetType());
if (dependencyProperty != null && dependencyObject != null)
{
if (dependencyObject.IsBindingSet(dependencyProperty))
targetObject = dependencyObject.GetBinding(dependencyProperty);
else
targetObject = dependencyObject.GetHandler(dependencyProperty);
}
else
targetObject = null;
//
eventArgs.Action = PathWalkAction.Stop;
break;
case PathMemberKind.Property:
memberType = (eventArgs.MemberInfo as PropertyInfo).PropertyType;
if (!(eventArgs.MemberInfo as PropertyInfo).CanRead)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
targetObject = (eventArgs.MemberInfo as PropertyInfo).GetValue(targetObject, null);
break;
case PathMemberKind.IndexedProperty:
memberType = (eventArgs.MemberInfo as PropertyInfo).PropertyType;
if (!(eventArgs.MemberInfo as PropertyInfo).CanRead)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
targetObject = (eventArgs.MemberInfo as PropertyInfo).GetValue(targetObject, eventArgs.IndexParameters);
break;
case PathMemberKind.Index://
memberType = (eventArgs.MemberInfo as PropertyInfo).PropertyType;
targetObject = (eventArgs.MemberInfo as PropertyInfo).GetValue(targetObject, BindingFlags.GetProperty, null, eventArgs.IndexParameters, CultureInfo.InvariantCulture);
break;
}
if (targetObject == null)
{
if (eventArgs.LastMemberInThePath)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
else
{
throw new InvalidOperationException(SR.GetString(SR.Error_BindPathNullValue, eventArgs.Path));
}
}
};
if (pathWalker.TryWalkPropertyPath(memberType, path))
{
//success
return ((targetObject != dataContext) ? targetObject : null);
}
else
{
//failure
return null;
}
}
internal static void SetValue(object dataContext, string path, object value)
{
if (dataContext == null)
throw new ArgumentNullException("dataContext");
if (string.IsNullOrEmpty(path))
throw new ArgumentNullException("path");
object parentObj = null;
object obj = dataContext;
object[] args = null;
MemberInfo memberInfo = null;
PathWalker pathWalker = new PathWalker();
pathWalker.MemberFound += delegate(object sender, PathMemberInfoEventArgs eventArgs)
{
//
if (obj == null)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
parentObj = obj;
memberInfo = eventArgs.MemberInfo;
switch (eventArgs.MemberKind)
{
case PathMemberKind.Field:
obj = (eventArgs.MemberInfo as FieldInfo).GetValue(parentObj);
args = null;
break;
case PathMemberKind.Event:
//
eventArgs.Action = PathWalkAction.Cancel; //set value is not supported on events
return;
case PathMemberKind.Property:
obj = (eventArgs.MemberInfo as PropertyInfo).GetValue(parentObj, null);
args = null;
break;
case PathMemberKind.IndexedProperty:
case PathMemberKind.Index:
obj = (eventArgs.MemberInfo as PropertyInfo).GetValue(parentObj, eventArgs.IndexParameters);
args = eventArgs.IndexParameters;
break;
}
};
if (pathWalker.TryWalkPropertyPath(dataContext.GetType(), path))
{
//at this point the 'obj' holds the old value, we will be changing it to 'value'
//success
if (memberInfo is FieldInfo)
{
(memberInfo as FieldInfo).SetValue(parentObj, value);
}
else if (memberInfo is PropertyInfo)
{
if ((memberInfo as PropertyInfo).CanWrite)
(memberInfo as PropertyInfo).SetValue(parentObj, value, args);
else
throw new InvalidOperationException(SR.GetString(SR.Error_ReadOnlyField, memberInfo.Name));
}
}
}
internal static ValidationError ValidateTypesInPath(Type srcType, string path)
{
ValidationError error = null;
if (srcType == null)
throw new ArgumentNullException("srcType");
if (path == null)
throw new ArgumentNullException("path");
if (path.Length == 0)
throw new ArgumentException(SR.GetString(SR.Error_EmptyPathValue), "path");
Debug.Assert(WorkflowCompilationContext.Current != null, "Can't have checkTypes set to true without a context in scope");
IList<AuthorizedType> authorizedTypes = WorkflowCompilationContext.Current.GetAuthorizedTypes();
if (authorizedTypes == null)
{
return new ValidationError(SR.GetString(SR.Error_ConfigFileMissingOrInvalid), ErrorNumbers.Error_ConfigFileMissingOrInvalid);
}
Type propertyType = srcType;
MemberInfo memberInfo = null;
PathWalker pathWalker = new PathWalker();
pathWalker.MemberFound += delegate(object sender, PathMemberInfoEventArgs eventArgs)
{
Type memberType = null;
memberInfo = eventArgs.MemberInfo;
if (memberInfo is FieldInfo)
memberType = ((FieldInfo)memberInfo).FieldType;
if (memberInfo is PropertyInfo)
memberType = ((PropertyInfo)memberInfo).PropertyType;
if (memberType != null && !SafeType(authorizedTypes, memberType))
{
error = new ValidationError(SR.GetString(SR.Error_TypeNotAuthorized, memberType), ErrorNumbers.Error_TypeNotAuthorized);
eventArgs.Action = PathWalkAction.Stop;
return;
}
};
pathWalker.TryWalkPropertyPath(propertyType, path);
return error;
}
private static bool SafeType(IList<AuthorizedType> authorizedTypes, Type referenceType)
{
bool authorized = false;
foreach (AuthorizedType authorizedType in authorizedTypes)
{
if (authorizedType.RegularExpression.IsMatch(referenceType.AssemblyQualifiedName))
{
authorized = (String.Compare(bool.TrueString, authorizedType.Authorized, StringComparison.OrdinalIgnoreCase) == 0);
if (!authorized)
return false;
}
}
return authorized;
}
internal static MemberInfo GetMemberInfo(Type srcType, string path)
{
if (srcType == null)
throw new ArgumentNullException("srcType");
if (path == null)
throw new ArgumentNullException("path");
if (path.Length == 0)
throw new ArgumentException(SR.GetString(SR.Error_EmptyPathValue), "path");
Type propertyType = srcType;
MemberInfo memberInfo = null;
PathWalker pathWalker = new PathWalker();
pathWalker.MemberFound += delegate(object sender, PathMemberInfoEventArgs eventArgs)
{
memberInfo = eventArgs.MemberInfo;
if (eventArgs.MemberKind == PathMemberKind.Event)
{
//need to exit!!!
eventArgs.Action = PathWalkAction.Stop;
return;
}
};
if (pathWalker.TryWalkPropertyPath(propertyType, path))
return memberInfo;
else
return null;
}
}
#endregion
#region FieldBind
[ActivityValidator(typeof(FieldBindValidator))]
internal sealed class FieldBind : MemberBind
{
private string path = string.Empty;
public FieldBind()
{
}
public FieldBind(string name)
: base(name)
{
}
public FieldBind(string name, string path)
: base(name)
{
this.path = path;
}
public string Path
{
get
{
return this.path;
}
set
{
if (!this.designMode)
throw new InvalidOperationException(SR.GetString(SR.Error_CanNotChangeAtRuntime));
this.path = value;
}
}
public override object GetRuntimeValue(Activity activity, Type targetType)
{
throw new NotImplementedException();
}
public override object GetRuntimeValue(Activity activity)
{
throw new NotImplementedException();
}
public override void SetRuntimeValue(Activity activity, object value)
{
throw new NotImplementedException();
}
protected override void OnRuntimeInitialized(Activity activity)
{
throw new NotImplementedException();
}
}
#endregion
#region PropertyBind
[ActivityValidator(typeof(PropertyBindValidator))]
internal sealed class PropertyBind : MemberBind
{
private string path = string.Empty;
public PropertyBind()
{
}
public PropertyBind(string name)
: base(name)
{
}
public PropertyBind(string name, string path)
: base(name)
{
this.path = path;
}
public string Path
{
get
{
return this.path;
}
set
{
if (!this.designMode)
throw new InvalidOperationException(SR.GetString(SR.Error_CanNotChangeAtRuntime));
this.path = value;
}
}
public override object GetRuntimeValue(Activity activity, Type targetType)
{
throw new NotImplementedException();
}
public override object GetRuntimeValue(Activity activity)
{
throw new NotImplementedException();
}
public override void SetRuntimeValue(Activity activity, object value)
{
throw new NotImplementedException();
}
}
#endregion
#region MethodBind
[ActivityValidator(typeof(MethodBindValidator))]
internal sealed class MethodBind : MemberBind
{
public MethodBind()
{
}
public MethodBind(string name)
: base(name)
{
}
public override object GetRuntimeValue(Activity activity, Type targetType)
{
throw new NotImplementedException();
}
public override object GetRuntimeValue(Activity activity)
{
throw new Exception(SR.GetString(SR.Error_NoTargetTypeForMethod));
}
public override void SetRuntimeValue(Activity activity, object value)
{
throw new Exception(SR.GetString(SR.Error_MethodDataSourceIsReadOnly));
}
}
#endregion
#endregion
#region ActivityBind
internal enum ActivityBindTypes { Field = 1, Property = 2, Method = 3 };
[Browsable(true)]
[TypeConverter(typeof(ActivityBindTypeConverter))]
[ActivityValidator(typeof(ActivityBindValidator))]
[DesignerSerializer(typeof(BindMarkupExtensionSerializer), typeof(WorkflowMarkupSerializer))]
[Obsolete("The System.Workflow.* types are deprecated. Instead, please use the new types from System.Activities.*")]
public sealed class ActivityBind : MarkupExtension, IPropertyValueProvider
{
#region stuff from the former Bind
[NonSerialized]
private bool designMode = true;
[NonSerialized]
private bool dynamicUpdateMode = false;
[NonSerialized]
private IDictionary userData = null;
[NonSerialized]
private object syncRoot = new object();
internal void SetContext(Activity activity)
{
this.designMode = false;
OnRuntimeInitialized(activity);
}
internal bool DynamicUpdateMode
{
get
{
return this.dynamicUpdateMode;
}
set
{
this.dynamicUpdateMode = false;
}
}
[Browsable(false)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
private bool DesignMode
{
get
{
return this.designMode && !this.dynamicUpdateMode;
}
}
[Browsable(false)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
public IDictionary UserData
{
get
{
if (this.userData == null)
{
lock (this.syncRoot)
{
if (this.userData == null)
this.userData = Hashtable.Synchronized(new Hashtable());
}
}
return this.userData;
}
}
internal static object GetDataSourceObject(Activity activity, string inputName, out string name)
{
if (activity == null)
throw new ArgumentNullException("activity");
if (string.IsNullOrEmpty(inputName))
throw new ArgumentNullException("inputName");
Activity contextActivity = Helpers.GetDataSourceActivity(activity, inputName, out name);
return contextActivity;
}
#endregion
private string id = string.Empty;
private string path = string.Empty;
public ActivityBind()
{
}
public ActivityBind(string name)
{
this.id = name;
}
public ActivityBind(string name, string path)
{
this.id = name;
this.path = path;
}
[DefaultValue("")]
[SRDescription(SR.ActivityBindIDDescription)]
[ConstructorArgument("name")]
public string Name
{
get
{
return this.id;
}
set
{
if (!this.DesignMode)
throw new InvalidOperationException(SR.GetString(SR.Error_CanNotChangeAtRuntime));
this.id = value;
}
}
[DefaultValue("")]
[SRDescription(SR.ActivityBindPathDescription)]
[TypeConverter(typeof(ActivityBindPathTypeConverter))]
public string Path
{
get
{
return this.path;
}
set
{
if (!this.DesignMode)
throw new InvalidOperationException(SR.GetString(SR.Error_CanNotChangeAtRuntime));
this.path = value;
}
}
public override object ProvideValue(IServiceProvider provider)
{
return this;
}
public object GetRuntimeValue(Activity activity, Type targetType)
{
if (activity == null)
throw new ArgumentNullException("activity");
if (targetType == null)
throw new ArgumentNullException("targetType");
return this.InternalGetRuntimeValue(activity, targetType);
}
public object GetRuntimeValue(Activity activity)
{
if (activity == null)
throw new ArgumentNullException("activity");
return this.InternalGetRuntimeValue(activity, null);
}
private object InternalGetRuntimeValue(Activity activity, Type targetType)
{
object runtimeValue = null;
Activity referencedActivity = Helpers.ParseActivityForBind(activity, this.Name);
if (referencedActivity != null)
{
//Now lets get the MemberInfo
MemberInfo memberInfo = ActivityBind.GetMemberInfo(referencedActivity.GetType(), Path, targetType);
if (memberInfo != null)
{
runtimeValue = ActivityBind.GetMemberValue(referencedActivity, memberInfo, Path, targetType);
if (runtimeValue is ActivityBind && BindHelpers.GetMemberType(memberInfo) != typeof(ActivityBind))
runtimeValue = ((ActivityBind)runtimeValue).GetRuntimeValue(referencedActivity, targetType);
}
else
{
// The value of this ActivityBind is bound to properties or events defined on the referenced activity
// Note that we don't have corresponding logic for SetRuntimeValue because value should be only set
// at the end of the activity reference chain.
Activity rootActivity = Helpers.GetRootActivity(activity);
DependencyProperty dependencyProperty = DependencyProperty.FromName(this.Path, rootActivity.GetType());
}
}
return runtimeValue;
}
public void SetRuntimeValue(Activity activity, object value)
{
if (activity == null)
throw new ArgumentNullException("activity");
Activity referencedActivity = Helpers.ParseActivityForBind(activity, this.Name);
if (referencedActivity != null)
{
MemberInfo memberInfo = ActivityBind.GetMemberInfo(referencedActivity.GetType(), Path, null);
if (memberInfo != null)
{
ActivityBind bind = ActivityBind.GetMemberValue(referencedActivity, memberInfo, Path, null) as ActivityBind;
if (bind != null)
bind.SetRuntimeValue(referencedActivity, value);
else
MemberBind.SetValue(referencedActivity, this.Path, value);
}
// Dependency property
/*else
{
Activity rootActivity = Helpers.GetRootActivity(activity);
DependencyProperty dependencyProperty = DependencyProperty.FromName(this.Path, rootActivity.GetType());
if (dependencyProperty != null)
{
referencedActivity.SetValue(dependencyProperty, value);
}
}*/
}
}
private void OnRuntimeInitialized(Activity activity)
{
Activity dataSourceActivity = null;
ActivityBind activityBind = ActivityBind.GetContextBind(this, activity, out dataSourceActivity);
if (activityBind != null && dataSourceActivity != null)
{
Type companionType = dataSourceActivity.GetType();
if (companionType != null)
{
MemberInfo memberInfo = ActivityBind.GetMemberInfo(companionType, activityBind.Path, null);
if (memberInfo != null)
{
if (memberInfo is FieldInfo || memberInfo is PropertyInfo || memberInfo is EventInfo)
{
if (activityBind.UserData[UserDataKeys.BindDataSource] == null)
activityBind.UserData[UserDataKeys.BindDataSource] = new Hashtable();
((Hashtable)activityBind.UserData[UserDataKeys.BindDataSource])[activity.QualifiedName] = memberInfo;
if (dataSourceActivity != null)
{
if (activityBind.UserData[UserDataKeys.BindDataContextActivity] == null)
activityBind.UserData[UserDataKeys.BindDataContextActivity] = new Hashtable();
((Hashtable)activityBind.UserData[UserDataKeys.BindDataContextActivity])[activity.QualifiedName] = dataSourceActivity.QualifiedName;
}
}
}
/*else
{
Activity rootActivity = Helpers.GetRootActivity(activity);
DependencyProperty dependencyProperty = DependencyProperty.FromName(activityBind.Path, rootActivity.GetType());
if (dependencyProperty != null)
{
if (activityBind.UserData[UserDataKeys.BindDataSource] == null)
activityBind.UserData[UserDataKeys.BindDataSource] = new Hashtable();
((Hashtable)activityBind.UserData[UserDataKeys.BindDataSource])[activity.QualifiedName] = dependencyProperty;
if (dataSourceActivity != null)
{
if (activityBind.UserData[UserDataKeys.BindDataContextActivity] == null)
activityBind.UserData[UserDataKeys.BindDataContextActivity] = new Hashtable();
((Hashtable)activityBind.UserData[UserDataKeys.BindDataContextActivity])[activity.QualifiedName] = dataSourceActivity.QualifiedName;
}
}
}*/
}
}
}
public override string ToString()
{
Activity activity = UserData[UserDataKeys.BindDataContextActivity] as Activity;
if (activity != null)
{
string bindString = String.Empty;
if (!string.IsNullOrEmpty(Name))
bindString = Helpers.ParseActivityForBind(activity, Name).QualifiedName;
if (!string.IsNullOrEmpty(Path))
{
string path = Path;
int indexOfSeparator = path.IndexOfAny(new char[] { '.', '/', '[' });
path = ((indexOfSeparator != -1)) ? path.Substring(0, indexOfSeparator) : path;
bindString += (!String.IsNullOrEmpty(bindString)) ? "." + path : path;
}
return bindString;
}
else
{
return base.ToString();
}
}
#region Runtime / Validation Time Helpers
internal static MemberInfo GetMemberInfo(Type dataSourceType, string path, Type targetType)
{
MemberInfo memberInfo = MemberBind.GetMemberInfo(dataSourceType, path);
//The events can be either bound to properties or can be bound to methods,
//There are cases where fields and methods can be of same name so in that case we either make sure for
//in the case of event handlers we either find Property or a Method
if (targetType != null && typeof(Delegate).IsAssignableFrom(targetType) && (memberInfo == null || !(memberInfo is EventInfo)))
{
MethodInfo delegateMethod = targetType.GetMethod("Invoke");
List<Type> paramTypes = new List<Type>();
foreach (ParameterInfo paramInfo in delegateMethod.GetParameters())
paramTypes.Add(paramInfo.ParameterType);
memberInfo = dataSourceType.GetMethod(path, BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static | BindingFlags.FlattenHierarchy, null, paramTypes.ToArray(), null);
}
return memberInfo;
}
private static object GetMemberValue(object dataSourceObject, MemberInfo memberInfo, string path, Type targetType)
{
object memberValue = null;
if (memberInfo is FieldInfo || memberInfo is PropertyInfo || memberInfo is EventInfo)
{
memberValue = MemberBind.GetValue(memberInfo, dataSourceObject, path);
/*if (memberValue != null && targetType != null &&
(memberValue.GetType().IsPrimitive || memberValue.GetType().IsEnum || memberValue.GetType() == typeof(string))
&& !targetType.IsAssignableFrom(memberValue.GetType()))
{
try
{
memberValue = Convert.ChangeType(memberValue, targetType, CultureInfo.InvariantCulture);
}
catch (Exception e)
{
throw new Exception(SR.GetString(SR.Error_DataSourceTypeConversionFailed, memberInfo.Name, memberValue.ToString(), targetType.FullName), e);
}
}*/
}
else if (targetType != null && memberInfo is MethodInfo)
{
memberValue = Delegate.CreateDelegate(targetType, dataSourceObject, (MethodInfo)memberInfo); //the wrapper method will never be static (even if the original one is)
}
else
{
throw new InvalidOperationException(SR.GetString(SR.Error_MemberNotFound));
}
return memberValue;
}
//This function is used to get the outermost activity bind which is bound to actual field/property
//The function is called in OnRuntimeInitialized and makes sure that we get outer target bind and cache
//it so that at runtime we do not have to walk the bind chain to find the bound member
private static ActivityBind GetContextBind(ActivityBind activityBind, Activity activity, out Activity contextActivity)
{
if (activityBind == null)
throw new ArgumentNullException("activityBind");
if (activity == null)
throw new ArgumentNullException("activity");
BindRecursionContext recursionContext = new BindRecursionContext();
ActivityBind contextBind = activityBind;
contextActivity = activity;
while (contextBind != null)
{
Activity resolvedActivity = Helpers.ParseActivityForBind(contextActivity, contextBind.Name);
if (resolvedActivity == null)
return null;
object dataSourceObject = resolvedActivity;
MemberInfo memberInfo = ActivityBind.GetMemberInfo(dataSourceObject.GetType(), contextBind.Path, null);
if (memberInfo == null)
{
contextActivity = resolvedActivity;
return contextBind;
}
else if (memberInfo is FieldInfo)
{
contextActivity = resolvedActivity;
return contextBind;
}
else if (memberInfo is PropertyInfo && (memberInfo as PropertyInfo).PropertyType == typeof(ActivityBind) && dataSourceObject != null)
{
object value = MemberBind.GetValue(memberInfo, dataSourceObject, contextBind.Path);
if (value is ActivityBind)
{
if (recursionContext.Contains(contextActivity, contextBind))
return null;
recursionContext.Add(contextActivity, contextBind);
contextActivity = resolvedActivity;
contextBind = value as ActivityBind;
}
else
{
return null;
}
}
else
{
return null;
}
}
return contextBind;
}
#endregion
#region DesignTime Integration (DO NOT CALL THESE AT RUNTIME)
#region Helper Functions
// This function replaces Activity1.code1 with /ParentContext.code1. This must be done for Binds that refer to
// to fields, properties and methods in the top level custom activity data context.
internal static string GetRelativePathExpression(Activity parentActivity, Activity childActivity)
{
string relativeBindExpression = String.Empty;
Activity rootActivity = Helpers.GetRootActivity(childActivity);
if (rootActivity == childActivity)
relativeBindExpression = "/Self";
else
relativeBindExpression = parentActivity.QualifiedName;
return relativeBindExpression;
}
#region IPropertyValueProvider Implementation
ICollection IPropertyValueProvider.GetPropertyValues(ITypeDescriptorContext context)
{
ArrayList values = new ArrayList();
if (string.Equals(context.PropertyDescriptor.Name, "Path", StringComparison.Ordinal) && !String.IsNullOrEmpty(Name) && context.PropertyDescriptor is ActivityBindPathPropertyDescriptor)
{
ITypeDescriptorContext outerPropertyContext = ((ActivityBindPathPropertyDescriptor)context.PropertyDescriptor).OuterPropertyContext;
if (outerPropertyContext != null)
{
Activity activity = PropertyDescriptorUtils.GetComponent(outerPropertyContext) as Activity;
if (activity != null)
{
Activity targetActivity = Helpers.ParseActivityForBind(activity, Name);
if (targetActivity != null)
{
foreach (MemberInfo memberInfo in ActivityBindPropertyDescriptor.GetBindableMembers(targetActivity, outerPropertyContext))
values.Add(memberInfo.Name);
}
}
}
}
return values;
}
#endregion
#endregion
#endregion
}
#endregion
#region BindRecursionContext
internal sealed class BindRecursionContext
{
private Hashtable activityBinds = new Hashtable();
public bool Contains(Activity activity, ActivityBind bind)
{
if (activity == null)
throw new ArgumentNullException("activity");
if (bind == null)
throw new ArgumentNullException("bind");
if (this.activityBinds[activity] != null)
{
List<ActivityBind> binds = this.activityBinds[activity] as List<ActivityBind>;
foreach (ActivityBind prevBind in binds)
{
if (prevBind.Path == bind.Path)
return true;
}
}
return false;
}
public void Add(Activity activity, ActivityBind bind)
{
if (activity == null)
throw new ArgumentNullException("activity");
if (bind == null)
throw new ArgumentNullException("bind");
if (this.activityBinds[activity] == null)
this.activityBinds[activity] = new List<ActivityBind>();
((List<ActivityBind>)this.activityBinds[activity]).Add(bind);
}
}
#endregion
#region BindHelpers
internal static class BindHelpers
{
internal static Type GetBaseType(IServiceProvider serviceProvider, PropertyValidationContext validationContext)
{
Type type = null;
if (validationContext.Property is PropertyInfo)
{
type = Helpers.GetBaseType(validationContext.Property as PropertyInfo, validationContext.PropertyOwner, serviceProvider);
}
else if (validationContext.Property is DependencyProperty)
{
//
DependencyProperty dependencyProperty = validationContext.Property as DependencyProperty;
if (dependencyProperty != null)
{
if (type == null)
{
IDynamicPropertyTypeProvider basetypeProvider = validationContext.PropertyOwner as IDynamicPropertyTypeProvider;
if (basetypeProvider != null)
type = basetypeProvider.GetPropertyType(serviceProvider, dependencyProperty.Name);
}
if (type == null)
type = dependencyProperty.PropertyType;
}
}
return type;
}
internal static AccessTypes GetAccessType(IServiceProvider serviceProvider, PropertyValidationContext validationContext)
{
AccessTypes accessType = AccessTypes.Read;
if (validationContext.Property is PropertyInfo)
{
accessType = Helpers.GetAccessType(validationContext.Property as PropertyInfo, validationContext.PropertyOwner, serviceProvider);
}
else if (validationContext.Property is DependencyProperty)
{
IDynamicPropertyTypeProvider basetypeProvider = validationContext.PropertyOwner as IDynamicPropertyTypeProvider;
if (basetypeProvider != null)
accessType = basetypeProvider.GetAccessType(serviceProvider, ((DependencyProperty)validationContext.Property).Name);
}
return accessType;
}
internal static object ResolveActivityPath(Activity refActivity, string path)
{
if (refActivity == null)
throw new ArgumentNullException("refActivity");
if (path == null)
throw new ArgumentNullException("path");
if (path.Length == 0)
throw new ArgumentException(SR.GetString(SR.Error_EmptyPathValue), "path");
object value = refActivity;
BindRecursionContext recursionContext = new BindRecursionContext();
PathWalker pathWalker = new PathWalker();
pathWalker.MemberFound += delegate(object sender, PathMemberInfoEventArgs eventArgs)
{
// If value is null, we don't want to use GetValue on the MemberInfo
if (value == null)
{
eventArgs.Action = PathWalkAction.Cancel; //need to cancel the walk with the failure return result
return;
}
switch (eventArgs.MemberKind)
{
case PathMemberKind.Field:
try
{
value = (eventArgs.MemberInfo as FieldInfo).GetValue(value);
}
catch (Exception exception)
{
//in some cases the value might not be there yet (e.g. validation vs. runtime)
value = null;
eventArgs.Action = PathWalkAction.Cancel;
//we should throw only if we are at the runtime
if (!refActivity.DesignMode)
{
TargetInvocationException targetInvocationException = exception as TargetInvocationException;
throw (targetInvocationException != null) ? targetInvocationException.InnerException : exception;
}
}
break;
case PathMemberKind.Event:
EventInfo evt = eventArgs.MemberInfo as EventInfo;
// GetValue() returns the actual value of the property. We need the Bind object here.
// Find out if there is a matching dependency property and get the value throw the DP.
DependencyProperty eventDependencyProperty = DependencyProperty.FromName(evt.Name, value.GetType());
if (eventDependencyProperty != null && value is DependencyObject)
{
if ((value as DependencyObject).IsBindingSet(eventDependencyProperty))
value = (value as DependencyObject).GetBinding(eventDependencyProperty);
else
value = (value as DependencyObject).GetHandler(eventDependencyProperty);
}
break;
case PathMemberKind.Property:
if (!(eventArgs.MemberInfo as PropertyInfo).CanRead)
{
eventArgs.Action = PathWalkAction.Cancel;
return;
}
// GetValue() returns the actual value of the property. We need the Bind object here.
// Find out if there is a matching dependency property and get the value throw the DP.
DependencyProperty dependencyProperty = DependencyProperty.FromName(eventArgs.MemberInfo.Name, value.GetType());
if (dependencyProperty != null && value is DependencyObject && (value as DependencyObject).IsBindingSet(dependencyProperty))
value = (value as DependencyObject).GetBinding(dependencyProperty);
else
try
{
value = (eventArgs.MemberInfo as PropertyInfo).GetValue(value, null);
}
catch (Exception exception)
{
//property getter function might throw at design time, validation should not fail bacause of that
value = null;
eventArgs.Action = PathWalkAction.Cancel;
//we should throw only if we are at the runtime
if (!refActivity.DesignMode)
{
TargetInvocationException targetInvocationException = exception as TargetInvocationException;
throw (targetInvocationException != null) ? targetInvocationException.InnerException : exception;
}
}
break;
case PathMemberKind.IndexedProperty:
case PathMemberKind.Index:
try
{
value = (eventArgs.MemberInfo as PropertyInfo).GetValue(value, BindingFlags.GetProperty, null, eventArgs.IndexParameters, CultureInfo.InvariantCulture);
}
catch (Exception exception)
{
//in some cases the value might not be there yet - e.g. array or dictionary is populated at runtime only (validation vs. runtime)
value = null;
eventArgs.Action = PathWalkAction.Cancel;
//we should throw only if we are at the runtime
if (!refActivity.DesignMode)
{
TargetInvocationException targetInvocationException = exception as TargetInvocationException;
throw (targetInvocationException != null) ? targetInvocationException.InnerException : exception;
}
}
break;
}
//need to unwrap the activity bind if we get one - to proceed with the actual field/property/delegate
//do not unwrap if the property/field is itself of type ActivityBind
//we should not unwrap the latest ActivityBind though - only intermediate ones
//avoid circular reference problems with the BindRecursionContext
if (value is ActivityBind && !eventArgs.LastMemberInThePath && GetMemberType(eventArgs.MemberInfo) != typeof(ActivityBind))
{
while (value is ActivityBind)
{
ActivityBind activityBind = value as ActivityBind;
if (recursionContext.Contains(refActivity, activityBind))
throw new InvalidOperationException(SR.GetString(SR.Bind_ActivityDataSourceRecursionDetected));
recursionContext.Add(refActivity, activityBind);
value = activityBind.GetRuntimeValue(refActivity);
}
}
};
if (pathWalker.TryWalkPropertyPath(refActivity.GetType(), path))
return value;
else
return null;
}
internal static PropertyInfo GetMatchedPropertyInfo(Type memberType, string[] aryArgName, object[] args)
{
if (memberType == null)
throw new ArgumentNullException("memberType");
if (aryArgName == null)
throw new ArgumentNullException("aryArgName");
if (args == null)
throw new ArgumentNullException("args");
MemberInfo[][] aryMembers = new MemberInfo[][] { memberType.GetDefaultMembers(), null };
if (memberType.IsArray)
{
MemberInfo[] getMember = memberType.GetMember("Get"); //arrays will always implement that
MemberInfo[] setMember = memberType.GetMember("Set"); //arrays will always implement that
PropertyInfo getProperty = new ActivityBindPropertyInfo(memberType, getMember[0] as MethodInfo, setMember[0] as MethodInfo, string.Empty, null);
aryMembers[1] = new MemberInfo[] { getProperty };
}
for (int index = 0; index < aryMembers.Length; ++index)
{
if (aryMembers[index] == null)
continue;
MemberInfo[] defaultMembers = aryMembers[index];
foreach (MemberInfo memberInfo in defaultMembers)
{
PropertyInfo propertyInfo = memberInfo as PropertyInfo;
if (propertyInfo != null)
{
if (MatchIndexerParameters(propertyInfo, aryArgName, args))
return propertyInfo;
}
}
}
return null;
}
internal static bool MatchIndexerParameters(PropertyInfo propertyInfo, string[] argNames, object[] args)
{
if (propertyInfo == null)
throw new ArgumentNullException("propertyInfo");
if (argNames == null)
throw new ArgumentNullException("argNames");
if (args == null)
throw new ArgumentNullException("args");
ParameterInfo[] aryPI = propertyInfo.GetIndexParameters();
if (aryPI.Length != argNames.Length)
return false;
for (int index = 0; index < args.Length; ++index)
{
Type paramType = aryPI[index].ParameterType;
if (paramType != typeof(String) && paramType != typeof(System.Int32))
return false;
try
{
object arg = null;
string argName = argNames[index].Trim();
if (paramType == typeof(String) && argName.StartsWith("\"", StringComparison.Ordinal) && argName.EndsWith("\"", StringComparison.Ordinal))
arg = argName.Substring(1, argName.Length - 2).Trim();
else if (paramType == typeof(System.Int32))
arg = Convert.ChangeType(argName, typeof(System.Int32), CultureInfo.InvariantCulture);
if (arg != null)
args.SetValue(arg, index);
else
return false;
}
catch
{
return false;
}
}
return true;
}
internal static Type GetMemberType(MemberInfo memberInfo)
{
FieldInfo fieldInfo = memberInfo as FieldInfo;
if (fieldInfo != null)
return fieldInfo.FieldType;
PropertyInfo propertyInfo = memberInfo as PropertyInfo;
if (propertyInfo != null)
{
if (propertyInfo.PropertyType != null)
return propertyInfo.PropertyType;
//sometimes need to get the property type off the getter method
MethodInfo getter = propertyInfo.GetGetMethod();
return getter.ReturnType;
}
EventInfo eventInfo = memberInfo as EventInfo;
if (eventInfo != null)
return eventInfo.EventHandlerType;
return null;
}
}
#endregion
#region Class PathWalker
internal enum PathMemberKind { Field, Event, Property, IndexedProperty, Index }
internal enum PathWalkAction { Continue, Stop, Cancel }; //stop returns true, while cancel returns false
internal class PathMemberInfoEventArgs : EventArgs
{
private string path;
private Type parentType;
private PathMemberKind memberKind;
private MemberInfo memberInfo;
private object[] indexParameters = new object[0]; //not empty for IndexedProperty and Index types
private PathWalkAction action = PathWalkAction.Continue;
private bool lastMemberInThePath = false;
public PathMemberInfoEventArgs(string path, Type parentType, MemberInfo memberInfo, PathMemberKind memberKind, bool lastMemberInThePath)
{
if (string.IsNullOrEmpty(path))
throw new ArgumentNullException("path");
if (parentType == null)
throw new ArgumentNullException("parentType");
if (memberInfo == null)
throw new ArgumentNullException("memberInfo");
this.path = path;
this.parentType = parentType;
this.memberInfo = memberInfo;
this.memberKind = memberKind;
this.lastMemberInThePath = lastMemberInThePath;
}
public PathMemberInfoEventArgs(string path, Type parentType, MemberInfo memberInfo, PathMemberKind memberKind, bool lastMemberInThePath, object[] indexParameters)
: this(path, parentType, memberInfo, memberKind, lastMemberInThePath)
{
this.indexParameters = indexParameters;
}
public string Path
{
get { return this.path; }
}
//public Type ParentType
//{
// get { return this.parentType; }
//}
public MemberInfo MemberInfo
{
get { return this.memberInfo; }
}
public PathMemberKind MemberKind
{
get { return this.memberKind; }
}
public object[] IndexParameters
{
get { return this.indexParameters; }
}
public bool LastMemberInThePath
{
get { return this.lastMemberInThePath; }
}
public PathWalkAction Action
{
get { return this.action; }
set { this.action = value; }
}
}
internal class PathErrorInfoEventArgs : EventArgs
{
private SourceValueInfo info;
private string currentPath;
public PathErrorInfoEventArgs(SourceValueInfo info, string currentPath)
{
if (currentPath == null)
throw new ArgumentNullException("currentPath");
this.info = info;
this.currentPath = currentPath;
}
//public SourceValueInfo Info
//{
// get { return this.info; }
//}
//public string CurrentPath
//{
// get { return this.currentPath; }
//}
}
//common path walker
//it is based off the property types and the PathParser results
//caller might keep a ref to the actual object and call Get/Set value on the members returned in the PathMemberInfoEventArgs
internal class PathWalker
{
public EventHandler<PathMemberInfoEventArgs> MemberFound; //on every member along the path
public EventHandler<PathErrorInfoEventArgs> PathErrorFound; //if there was an error parsing or walking the path
private static MemberInfo[] PopulateMembers(Type type, string memberName)
{
List<MemberInfo> members = new List<MemberInfo>();
members.AddRange(type.GetMember(memberName, MemberTypes.Field | MemberTypes.Property | MemberTypes.Event | MemberTypes.Method, BindingFlags.Public | BindingFlags.Static | BindingFlags.Instance | BindingFlags.FlattenHierarchy));
if (type.IsInterface)
{
Type[] interfaces = type.GetInterfaces();
foreach (Type implementedInterface in interfaces)
{
members.AddRange(implementedInterface.GetMember(memberName, MemberTypes.Field | MemberTypes.Property | MemberTypes.Event | MemberTypes.Method, BindingFlags.Public | BindingFlags.Static | BindingFlags.Instance | BindingFlags.FlattenHierarchy));
}
}
return members.ToArray();
}
public bool TryWalkPropertyPath(Type rootType, string path)
{
if (rootType == null)
throw new ArgumentNullException("rootType");
if (string.IsNullOrEmpty(path))
throw new ArgumentNullException("path");
Type propertyType = rootType;
string currentPath = string.Empty;
PathParser parser = new PathParser();
List<SourceValueInfo> pathInfo = parser.Parse(path, true);
string parsingError = parser.Error;
for (int i = 0; i < pathInfo.Count; i++)
{
SourceValueInfo info = pathInfo[i];
if (string.IsNullOrEmpty(info.name))
{
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
string additionalPath = (info.type == SourceValueType.Property) ? info.name : "[" + info.name + "]";
string newPath = (string.IsNullOrEmpty(currentPath)) ? additionalPath : currentPath + ((info.type == SourceValueType.Property) ? "." : string.Empty) + additionalPath;
Type newPropertyType = null;
MemberInfo newMemberInfo = null;
switch (info.type)
{
case SourceValueType.Property:
MemberInfo[] members = PopulateMembers(propertyType, info.name);
if (members == null || members.Length == 0 || members[0] == null)
{
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
newMemberInfo = members[0];
if (newMemberInfo is EventInfo || newMemberInfo is MethodInfo)
{
if (MemberFound != null)
{
PathMemberInfoEventArgs args = new PathMemberInfoEventArgs(newPath, propertyType, newMemberInfo, PathMemberKind.Event, i == pathInfo.Count - 1);
MemberFound(this, args);
if (args.Action == PathWalkAction.Cancel)
return false;
else if (args.Action == PathWalkAction.Stop)
return true;
}
//
return string.IsNullOrEmpty(parsingError);
}
else if (newMemberInfo is PropertyInfo)
{
//property getter could be an indexer
PropertyInfo memberPropertyInfo = newMemberInfo as PropertyInfo;
MethodInfo getterMethod = memberPropertyInfo.GetGetMethod();
MethodInfo setterMethod = memberPropertyInfo.GetSetMethod();
ActivityBindPropertyInfo properyInfo = new ActivityBindPropertyInfo(propertyType, getterMethod, setterMethod, memberPropertyInfo.Name, memberPropertyInfo);
newPropertyType = properyInfo.PropertyType;
ParameterInfo[] parameters = properyInfo.GetIndexParameters();
if (parameters.Length > 0)
{
//need to check that the next parsed element is an indexer
if (i < pathInfo.Count - 1 && pathInfo[i + 1].type == SourceValueType.Indexer && !string.IsNullOrEmpty(pathInfo[i + 1].name))
{
string[] arrayArgName = pathInfo[i + 1].name.Split(',');
object[] arguments = new object[arrayArgName.Length];
//match the number/type of parameters from the following indexer item
if (BindHelpers.MatchIndexerParameters(properyInfo, arrayArgName, arguments))
{
newPath += "[" + pathInfo[i + 1].name + "]";
//indexer property, uses two of the parsed array elements
if (MemberFound != null)
{
PathMemberInfoEventArgs args = new PathMemberInfoEventArgs(newPath, propertyType, properyInfo, PathMemberKind.IndexedProperty, i == pathInfo.Count - 2, arguments);
MemberFound(this, args);
if (args.Action == PathWalkAction.Cancel)
return false;
else if (args.Action == PathWalkAction.Stop)
return true;
}
//skip the next indexer item too
i++;
}
else
{
//parameters didn't match
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
}
else
{
//trailing index is missing or empty for the indexed property
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
}
else // parameters.Length == 0
{
//a regular property
if (MemberFound != null)
{
PathMemberInfoEventArgs args = new PathMemberInfoEventArgs(newPath, propertyType, properyInfo, PathMemberKind.Property, i == pathInfo.Count - 1);
MemberFound(this, args);
if (args.Action == PathWalkAction.Cancel)
return false;
else if (args.Action == PathWalkAction.Stop)
return true;
}
}
}
else
{
//that would be a field
if (MemberFound != null)
{
PathMemberInfoEventArgs args = new PathMemberInfoEventArgs(newPath, propertyType, newMemberInfo, PathMemberKind.Field, i == pathInfo.Count - 1);
MemberFound(this, args);
if (args.Action == PathWalkAction.Cancel)
return false;
else if (args.Action == PathWalkAction.Stop)
return true;
}
newPropertyType = (newMemberInfo as FieldInfo).FieldType;
}
break;
case SourceValueType.Indexer:
if (!string.IsNullOrEmpty(info.name))
{
string[] arrayArgName = info.name.Split(',');
object[] arguments = new object[arrayArgName.Length];
PropertyInfo arrayPropertyInfo = BindHelpers.GetMatchedPropertyInfo(propertyType, arrayArgName, arguments);
if (arrayPropertyInfo != null)
{
if (MemberFound != null)
{
PathMemberInfoEventArgs args = new PathMemberInfoEventArgs(newPath, propertyType, arrayPropertyInfo, PathMemberKind.Index, i == pathInfo.Count - 1, arguments);
MemberFound(this, args);
if (args.Action == PathWalkAction.Cancel)
return false;
else if (args.Action == PathWalkAction.Stop)
return true;
}
newPropertyType = arrayPropertyInfo.PropertyType;
if (newPropertyType == null)
newPropertyType = arrayPropertyInfo.GetGetMethod().ReturnType;
}
else
{
//did not mach number/type of arguments
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
}
else
{
//empty indexer
if (PathErrorFound != null)
PathErrorFound(this, new PathErrorInfoEventArgs(info, currentPath));
return false;
}
break;
}
propertyType = newPropertyType;
currentPath = newPath;
}
return string.IsNullOrEmpty(parsingError);
}
}
//multi-dimentional arrays dont implement default property
//this fake property will help us to handle that
internal class ActivityBindPropertyInfo : PropertyInfo
{
private MethodInfo getMethod;
private MethodInfo setMethod;
private Type declaringType;
private string propertyName;
private PropertyInfo originalPropertyInfo; //in vb fields get returned as properties
public ActivityBindPropertyInfo(Type declaringType, MethodInfo getMethod, MethodInfo setMethod, string propertyName, PropertyInfo originalPropertyInfo)
{
if (declaringType == null)
throw new ArgumentNullException("declaringType");
if (propertyName == null)
throw new ArgumentNullException("propertyName");
this.declaringType = declaringType;
this.getMethod = getMethod;
this.setMethod = setMethod;
this.propertyName = propertyName;
this.originalPropertyInfo = originalPropertyInfo; //could be null for array indexers.
}
public override string Name
{
get { return this.propertyName; }
}
public override MethodInfo GetGetMethod(bool nonPublic)
{
return this.getMethod;
}
public override MethodInfo GetSetMethod(bool nonPublic)
{
return this.setMethod;
}
public override Type PropertyType
{
get
{
if (this.getMethod != null)
return this.getMethod.ReturnType;
else if (this.originalPropertyInfo != null)
return this.originalPropertyInfo.PropertyType;
else
return typeof(object);
}
}
public override ParameterInfo[] GetIndexParameters()
{
if (this.getMethod != null)
return this.getMethod.GetParameters();
else if (this.originalPropertyInfo != null)
return this.originalPropertyInfo.GetIndexParameters();
else
return new ParameterInfo[0];
}
public override object GetValue(object obj, BindingFlags invokeAttr, Binder binder, object[] index, CultureInfo culture)
{
if (this.getMethod == null && (this.originalPropertyInfo == null || !this.originalPropertyInfo.CanRead))
throw new InvalidOperationException(SR.GetString(SR.Error_PropertyHasNoGetterDefined, this.propertyName));
if (this.getMethod != null)
return this.getMethod.Invoke(obj, invokeAttr, binder, index, culture);
else
return this.originalPropertyInfo.GetValue(obj, invokeAttr, binder, index, culture);
}
public override void SetValue(object obj, object value, BindingFlags invokeAttr, Binder binder, object[] index, CultureInfo culture)
{
if (this.setMethod == null && (this.originalPropertyInfo == null || !this.originalPropertyInfo.CanWrite))
throw new InvalidOperationException(SR.GetString(SR.Error_PropertyHasNoSetterDefined, this.propertyName));
if (this.setMethod != null)
{
object[] parameters = new object[((index != null) ? index.Length : 0) + 1];
parameters[((index != null) ? index.Length : 0)] = value;
if (index != null)
index.CopyTo(parameters, 0);
this.setMethod.Invoke(obj, invokeAttr, binder, parameters, culture);
}
else
{
this.originalPropertyInfo.SetValue(obj, value, invokeAttr, binder, index, culture);
}
}
public override MethodInfo[] GetAccessors(bool nonPublic)
{
return new MethodInfo[] { this.getMethod, this.setMethod };
}
public override PropertyAttributes Attributes
{
get { return PropertyAttributes.None; }
}
public override bool CanRead
{
get
{
if (this.getMethod != null)
return true;
else if (this.originalPropertyInfo != null)
return this.originalPropertyInfo.CanRead;
else
return false;
}
}
public override bool CanWrite
{
get
{
if (this.setMethod != null)
return true;
else if (this.originalPropertyInfo != null)
return this.originalPropertyInfo.CanWrite;
else
return false;
}
}
public override Type DeclaringType
{
get { return this.declaringType; }
}
public override Type ReflectedType
{
get { return this.declaringType; }
}
public override object[] GetCustomAttributes(bool inherit)
{
return new object[0];
}
public override object[] GetCustomAttributes(Type attributeType, bool inherit)
{
return new object[0];
}
public override bool IsDefined(Type attributeType, bool inherit)
{
throw new NotSupportedException();
}
}
#endregion
#region PathParser
internal enum SourceValueType
{
Property,
Indexer
};
internal enum DrillIn
{
Never,
IfNeeded
};
internal struct SourceValueInfo
{
internal SourceValueType type;
internal DrillIn drillIn;
internal string name;
internal SourceValueInfo(SourceValueType t, DrillIn d, string n)
{
type = t;
drillIn = d;
name = n;
}
}
internal sealed class PathParser
{
private string error = string.Empty;
private State state;
private string pathValue;
private int index;
private int pathLength;
private DrillIn drillIn;
private List<SourceValueInfo> al = new List<SourceValueInfo>();
private const char NullChar = Char.MinValue;
private static List<SourceValueInfo> EmptyInfo = new List<SourceValueInfo>(1);
private static string SpecialChars = @".[]";
private enum State
{
Init,
Prop,
Done
};
internal String Error
{
get
{
return error;
}
}
internal List<SourceValueInfo> Parse(string path, bool returnResultBeforeError)
{
this.pathValue = (path != null) ? path.Trim() : String.Empty;
this.pathLength = this.pathValue.Length;
this.index = 0;
this.drillIn = DrillIn.IfNeeded;
this.al.Clear();
this.error = null;
this.state = State.Init;
if (this.pathLength > 0 && this.pathValue[0] == '.')
{
//empty first prop - > input path was like ".bar"; need to add first empty property
SourceValueInfo info = new SourceValueInfo(SourceValueType.Property, this.drillIn, string.Empty);
this.al.Add(info);
}
while (this.state != State.Done)
{
char c = (this.index < this.pathLength) ? this.pathValue[this.index] : NullChar;
switch (this.state)
{
case State.Init:
switch (c)
{
case '/':
case '.':
case '[':
case NullChar:
this.state = State.Prop;
break;
case ']'://unexpected close indexer, report error
this.error = "path[" + this.index + "] = " + c;
return returnResultBeforeError ? this.al : EmptyInfo;
default:
AddProperty();
break;
}
break;
case State.Prop:
bool isIndexer = false;
switch (c)
{
case '.':
this.drillIn = DrillIn.Never;
break;
case '[':
isIndexer = true;
break;
case NullChar:
--this.index;
break;
default:
this.error = "path[" + this.index + "] = " + c;
return returnResultBeforeError ? this.al : EmptyInfo;
}
++this.index; // skip over special character
if (isIndexer)
AddIndexer();
else
AddProperty();
break;
}
}
return (this.error == null || returnResultBeforeError) ? this.al : EmptyInfo;
}
private void AddProperty()
{
int start = this.index;
while (this.index < this.pathLength && SpecialChars.IndexOf(this.pathValue[this.index]) < 0)
++this.index;
string name = this.pathValue.Substring(start, this.index - start).Trim();
SourceValueInfo info = new SourceValueInfo(
SourceValueType.Property,
this.drillIn, name);
this.al.Add(info);
StartNewLevel();
}
private void AddIndexer()
{
int start = this.index;
int level = 1;
while (level > 0)
{
if (this.index >= this.pathLength)
{
return;
}
if (this.pathValue[this.index] == '[')
++level;
else if (this.pathValue[this.index] == ']')
--level;
++this.index;
}
string name = this.pathValue.Substring(start, this.index - start - 1).Trim();
SourceValueInfo info = new SourceValueInfo(
SourceValueType.Indexer,
this.drillIn, name);
this.al.Add(info);
StartNewLevel();
}
private void StartNewLevel()
{
if (this.index >= this.pathLength)
this.state = State.Done;
this.drillIn = DrillIn.Never;
}
}
#endregion
}