You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			223 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			223 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //----------------------------------------------------------------------------- | |||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | |||
|  | //----------------------------------------------------------------------------- | |||
|  | 
 | |||
|  | namespace System.Activities.Expressions | |||
|  | { | |||
|  |     using System.Collections.ObjectModel; | |||
|  |     using System.ComponentModel; | |||
|  |     using System.Reflection; | |||
|  |     using System.Runtime; | |||
|  |     using System.Runtime.Collections; | |||
|  |     using System.Runtime.Serialization; | |||
|  |     using System.Windows.Markup; | |||
|  |     using System.Threading; | |||
|  | 
 | |||
|  |     [ContentProperty("Indices")] | |||
|  |     public sealed class IndexerReference<TOperand, TItem> : CodeActivity<Location<TItem>> | |||
|  |     { | |||
|  |         Collection<InArgument> indices; | |||
|  |         MethodInfo getMethod; | |||
|  |         MethodInfo setMethod; | |||
|  |         Func<object, object[], object> getFunc; | |||
|  |         Func<object, object[], object> setFunc; | |||
|  | 
 | |||
|  |         static MruCache<MethodInfo, Func<object, object[], object>> funcCache = | |||
|  |             new MruCache<MethodInfo, Func<object, object[], object>>(MethodCallExpressionHelper.FuncCacheCapacity); | |||
|  |         static ReaderWriterLockSlim locker = new ReaderWriterLockSlim(); | |||
|  | 
 | |||
|  |         [RequiredArgument] | |||
|  |         [DefaultValue(null)] | |||
|  |         public InArgument<TOperand> Operand | |||
|  |         { | |||
|  |             get; | |||
|  |             set; | |||
|  |         } | |||
|  | 
 | |||
|  |         [RequiredArgument] | |||
|  |         [DefaultValue(null)] | |||
|  |         public Collection<InArgument> Indices | |||
|  |         { | |||
|  |             get | |||
|  |             { | |||
|  |                 if (this.indices == null) | |||
|  |                 { | |||
|  |                     this.indices = new ValidatingCollection<InArgument> | |||
|  |                     {    | |||
|  |                         // disallow null values | |||
|  |                         OnAddValidationCallback = item => | |||
|  |                         { | |||
|  |                             if (item == null) | |||
|  |                             { | |||
|  |                                 throw FxTrace.Exception.ArgumentNull("item"); | |||
|  |                             } | |||
|  |                         }, | |||
|  |                     }; | |||
|  |                 } | |||
|  |                 return this.indices; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         protected override void CacheMetadata(CodeActivityMetadata metadata) | |||
|  |         { | |||
|  |             MethodInfo oldGetMethod = this.getMethod; | |||
|  |             MethodInfo oldSetMethod = this.setMethod; | |||
|  | 
 | |||
|  |             if (typeof(TOperand).IsValueType) | |||
|  |             { | |||
|  |                 metadata.AddValidationError(SR.TargetTypeIsValueType(this.GetType().Name, this.DisplayName)); | |||
|  |             } | |||
|  |             if (this.Indices.Count == 0) | |||
|  |             { | |||
|  |                 metadata.AddValidationError(SR.IndicesAreNeeded(this.GetType().Name, this.DisplayName)); | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 IndexerHelper.CacheMethod<TOperand, TItem>(this.Indices, ref this.getMethod, ref this.setMethod); | |||
|  |                 if (this.setMethod == null) | |||
|  |                 { | |||
|  |                     metadata.AddValidationError(SR.SpecialMethodNotFound("set_Item", typeof(TOperand).Name)); | |||
|  |                 } | |||
|  |             } | |||
|  |             | |||
|  |             RuntimeArgument operandArgument = new RuntimeArgument("Operand", typeof(TOperand), ArgumentDirection.In, true); | |||
|  |             metadata.Bind(this.Operand, operandArgument); | |||
|  |             metadata.AddArgument(operandArgument); | |||
|  |              | |||
|  |             IndexerHelper.OnGetArguments<TItem>(this.Indices, this.Result, metadata); | |||
|  |              if (MethodCallExpressionHelper.NeedRetrieve(this.getMethod, oldGetMethod, this.getFunc)) | |||
|  |             { | |||
|  |                 this.getFunc = MethodCallExpressionHelper.GetFunc(metadata, this.getMethod, funcCache, locker); | |||
|  |             } | |||
|  |             if (MethodCallExpressionHelper.NeedRetrieve(this.setMethod, oldSetMethod, this.setFunc)) | |||
|  |             { | |||
|  |                 this.setFunc = MethodCallExpressionHelper.GetFunc(metadata, this.setMethod, funcCache, locker); | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         protected override Location<TItem> Execute(CodeActivityContext context) | |||
|  |         { | |||
|  |             object[] indicesValue = new object[this.Indices.Count]; | |||
|  | 
 | |||
|  |             for (int i = 0; i < this.Indices.Count; i++) | |||
|  |             { | |||
|  |                 indicesValue[i] = this.Indices[i].Get(context); | |||
|  |             } | |||
|  | 
 | |||
|  |             TOperand operandValue = this.Operand.Get(context); | |||
|  |             if (operandValue == null) | |||
|  |             { | |||
|  |                 throw FxTrace.Exception.AsError(new InvalidOperationException(SR.MemberCannotBeNull("Operand", this.GetType().Name, this.DisplayName))); | |||
|  |             } | |||
|  | 
 | |||
|  |             return new IndexerLocation(operandValue, indicesValue, this.getMethod, this.setMethod, this.getFunc, this.setFunc); | |||
|  |         } | |||
|  | 
 | |||
|  |          | |||
|  |         [DataContract] | |||
|  |         internal class IndexerLocation : Location<TItem> | |||
|  |         { | |||
|  |             TOperand operand; | |||
|  | 
 | |||
|  |             object[] indices; | |||
|  | 
 | |||
|  |             object[] parameters; | |||
|  | 
 | |||
|  |             MethodInfo getMethod; | |||
|  | 
 | |||
|  |             MethodInfo setMethod; | |||
|  | 
 | |||
|  |             Func<object, object[], object> getFunc; | |||
|  |             Func<object, object[], object> setFunc; | |||
|  | 
 | |||
|  |             public IndexerLocation(TOperand operand, object[] indices, MethodInfo getMethod, MethodInfo setMethod,  | |||
|  |                 Func<object, object[], object> getFunc, Func<object, object[], object> setFunc) | |||
|  |                 : base() | |||
|  |             { | |||
|  |                 this.operand = operand; | |||
|  |                 this.indices = indices; | |||
|  |                 this.getMethod = getMethod; | |||
|  |                 this.setMethod = setMethod; | |||
|  |                 this.getFunc = getFunc; | |||
|  |                 this.setFunc = setFunc; | |||
|  |             } | |||
|  | 
 | |||
|  |             public override TItem Value | |||
|  |             { | |||
|  |                 get | |||
|  |                 { | |||
|  |                     Fx.Assert(this.operand != null, "operand must not be null"); | |||
|  |                     Fx.Assert(this.indices != null, "indices must not be null"); | |||
|  |                     if (this.getFunc != null) | |||
|  |                     { | |||
|  |                         return (TItem)this.getFunc(this.operand, indices); | |||
|  |                     } | |||
|  |                     else if (this.getMethod != null) | |||
|  |                     { | |||
|  |                         return (TItem)this.getMethod.Invoke(this.operand, indices); | |||
|  |                     } | |||
|  |                     throw FxTrace.Exception.AsError(new InvalidOperationException(SR.SpecialMethodNotFound("get_Item", typeof(TOperand).Name))); | |||
|  |                 } | |||
|  |                 set | |||
|  |                 { | |||
|  |                     Fx.Assert(this.setMethod != null, "setMethod must not be null"); | |||
|  |                     Fx.Assert(this.operand != null, "operand must not be null"); | |||
|  |                     Fx.Assert(this.indices != null, "indices must not be null"); | |||
|  |                     if (this.parameters == null) | |||
|  |                     { | |||
|  |                         this.parameters = new object[this.indices.Length + 1]; | |||
|  |                         for (int i = 0; i < this.indices.Length; i++) | |||
|  |                         { | |||
|  |                             parameters[i] = this.indices[i]; | |||
|  |                         } | |||
|  |                         parameters[parameters.Length - 1] = value; | |||
|  |                     } | |||
|  |                     if (this.setFunc != null) | |||
|  |                     { | |||
|  |                         this.setFunc(operand, parameters); | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         this.setMethod.Invoke(operand, parameters); | |||
|  |                     } | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             [DataMember(EmitDefaultValue = false, Name = "operand")] | |||
|  |             internal TOperand SerializedOperand | |||
|  |             { | |||
|  |                 get { return this.operand; } | |||
|  |                 set { this.operand = value; } | |||
|  |             } | |||
|  | 
 | |||
|  |             [DataMember(EmitDefaultValue = false, Name = "indices")] | |||
|  |             internal object[] SerializedIndices | |||
|  |             { | |||
|  |                 get { return this.indices; } | |||
|  |                 set { this.indices = value; } | |||
|  |             } | |||
|  | 
 | |||
|  |             [DataMember(EmitDefaultValue = false, Name = "parameters")] | |||
|  |             internal object[] SerializedParameters | |||
|  |             { | |||
|  |                 get { return this.parameters; } | |||
|  |                 set { this.parameters = value; } | |||
|  |             } | |||
|  | 
 | |||
|  |             [DataMember(EmitDefaultValue = false, Name = "getMethod")] | |||
|  |             internal MethodInfo SerializedGetMethod | |||
|  |             { | |||
|  |                 get { return this.getMethod; } | |||
|  |                 set { this.getMethod = value; } | |||
|  |             } | |||
|  | 
 | |||
|  |             [DataMember(EmitDefaultValue = false, Name = "setMethod")] | |||
|  |             internal MethodInfo SerializedSetMethod | |||
|  |             { | |||
|  |                 get { return this.setMethod; } | |||
|  |                 set { this.setMethod = value; } | |||
|  |             } | |||
|  |         } | |||
|  |     } | |||
|  | } |