You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			233 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			233 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | using System.Collections.Generic; | |||
|  | using System.ComponentModel.DataAnnotations.Resources; | |||
|  | using System.Diagnostics; | |||
|  | using System.Diagnostics.CodeAnalysis; | |||
|  | using System.Globalization; | |||
|  | using System.Linq; | |||
|  | 
 | |||
|  | namespace System.ComponentModel.DataAnnotations { | |||
|  |     /// <summary> | |||
|  |     /// Attribute to provide a hint to the presentation layer about what control it should use | |||
|  |     /// </summary> | |||
|  |     [SuppressMessage("Microsoft.Design", "CA1019:DefineAccessorsForAttributeArguments", Justification = "ControlParameters is exposed, just with a different type")] | |||
|  |     [AttributeUsage(AttributeTargets.Property | AttributeTargets.Field, AllowMultiple = true)] | |||
|  |     [SuppressMessage("Microsoft.Performance", "CA1813:AvoidUnsealedAttributes", Justification = "We want users to be able to extend this class")] | |||
|  |     public class UIHintAttribute : Attribute { | |||
|  |         private UIHintImplementation _implementation; | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Gets the name of the control that is most appropriate for this associated property or field | |||
|  |         /// </summary> | |||
|  |         public string UIHint { | |||
|  |             get { | |||
|  |                 return this._implementation.UIHint; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Gets the name of the presentation layer that supports the control type in <see cref="UIHint"/> | |||
|  |         /// </summary> | |||
|  |         public string PresentationLayer { | |||
|  |             get { | |||
|  |                 return this._implementation.PresentationLayer; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Gets the name-value pairs used as parameters to the control's constructor | |||
|  |         /// </summary> | |||
|  |         /// <exception cref="InvalidOperationException"> is thrown if the current attribute is ill-formed.</exception> | |||
|  |         public IDictionary<string, object> ControlParameters { | |||
|  |             get { | |||
|  |                 return this._implementation.ControlParameters; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  | #if !SILVERLIGHT | |||
|  |         /// <summary> | |||
|  |         /// Gets a unique identifier for this attribute. | |||
|  |         /// </summary> | |||
|  |         public override object TypeId { | |||
|  |             get { | |||
|  |                 return this; | |||
|  |             } | |||
|  |         } | |||
|  | #endif | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Constructor that accepts the name of the control, without specifying which presentation layer to use | |||
|  |         /// </summary> | |||
|  |         /// <param name="uiHint">The name of the UI control.</param> | |||
|  |         public UIHintAttribute(string uiHint) | |||
|  |             : this(uiHint, null, new object[0]) { | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Constructor that accepts both the name of the control as well as the presentation layer | |||
|  |         /// </summary> | |||
|  |         /// <param name="uiHint">The name of the control to use</param> | |||
|  |         /// <param name="presentationLayer">The name of the presentation layer that supports this control</param> | |||
|  |         public UIHintAttribute(string uiHint, string presentationLayer) | |||
|  |             : this(uiHint, presentationLayer, new object[0]) { | |||
|  |         } | |||
|  | 
 | |||
|  |         /// <summary> | |||
|  |         /// Full constructor that accepts the name of the control, presentation layer, and optional parameters | |||
|  |         /// to use when constructing the control | |||
|  |         /// </summary> | |||
|  |         /// <param name="uiHint">The name of the control</param> | |||
|  |         /// <param name="presentationLayer">The presentation layer</param> | |||
|  |         /// <param name="controlParameters">The list of parameters for the control</param> | |||
|  |         public UIHintAttribute(string uiHint, string presentationLayer, params object[] controlParameters) { | |||
|  |             this._implementation = new UIHintImplementation(uiHint, presentationLayer, controlParameters); | |||
|  |         } | |||
|  | 
 | |||
|  |         public override int GetHashCode() { | |||
|  |             return this._implementation.GetHashCode(); | |||
|  |         } | |||
|  | 
 | |||
|  |         public override bool Equals(object obj) { | |||
|  |             var otherAttribute = obj as UIHintAttribute; | |||
|  |             if (otherAttribute == null) { | |||
|  |                 return false; | |||
|  |             } | |||
|  |             return this._implementation.Equals(otherAttribute._implementation); | |||
|  |         } | |||
|  | 
 | |||
|  |         internal class UIHintImplementation { | |||
|  |             private IDictionary<string, object> _controlParameters; | |||
|  |             private object[] _inputControlParameters; | |||
|  | 
 | |||
|  |             /// <summary> | |||
|  |             /// Gets the name of the control that is most appropriate for this associated property or field | |||
|  |             /// </summary> | |||
|  |             public string UIHint { get; private set; } | |||
|  | 
 | |||
|  |             /// <summary> | |||
|  |             /// Gets the name of the presentation layer that supports the control type in <see cref="UIHint"/> | |||
|  |             /// </summary> | |||
|  |             public string PresentationLayer { get; private set; } | |||
|  | 
 | |||
|  |             public IDictionary<string, object> ControlParameters { | |||
|  |                 get { | |||
|  |                     if (this._controlParameters == null) { | |||
|  |                         // Lazy load the dictionary. It's fine if this method executes multiple times in stress scenarios. | |||
|  |                         // If the method throws (indicating that the input params are invalid) this property will throw | |||
|  |                         // every time it's accessed. | |||
|  |                         this._controlParameters = this.BuildControlParametersDictionary(); | |||
|  |                     } | |||
|  |                     return this._controlParameters; | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             public UIHintImplementation(string uiHint, string presentationLayer, params object[] controlParameters) { | |||
|  |                 this.UIHint = uiHint; | |||
|  |                 this.PresentationLayer = presentationLayer; | |||
|  |                 if (controlParameters != null) { | |||
|  |                     this._inputControlParameters = new object[controlParameters.Length]; | |||
|  |                     Array.Copy(controlParameters, this._inputControlParameters, controlParameters.Length); | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             /// <summary> | |||
|  |             /// Returns the hash code for this UIHintAttribute. | |||
|  |             /// </summary> | |||
|  |             /// <returns>A 32-bit signed integer hash code.</returns> | |||
|  |             public override int GetHashCode() { | |||
|  |                 var a = this.UIHint ?? String.Empty; | |||
|  |                 var b = this.PresentationLayer ?? String.Empty; | |||
|  | 
 | |||
|  |                 return a.GetHashCode() ^ b.GetHashCode(); | |||
|  |             } | |||
|  | 
 | |||
|  |             /// <summary> | |||
|  |             /// Determines whether this instance of UIHintAttribute and a specified object, | |||
|  |             /// which must also be a UIHintAttribute object, have the same value. | |||
|  |             /// </summary> | |||
|  |             /// <param name="obj">An System.Object.</param> | |||
|  |             /// <returns>true if obj is a UIHintAttribute and its value is the same as this instance; otherwise, false.</returns> | |||
|  |             public override bool Equals(object obj) { | |||
|  |                 // don't need to perform a type check on obj since this is an internal class | |||
|  |                 var otherImplementation = (UIHintImplementation)obj; | |||
|  | 
 | |||
|  |                 if (this.UIHint != otherImplementation.UIHint || this.PresentationLayer != otherImplementation.PresentationLayer) { | |||
|  |                     return false; | |||
|  |                 } | |||
|  | 
 | |||
|  |                 IDictionary<string, object> leftParams; | |||
|  |                 IDictionary<string, object> rightParams; | |||
|  | 
 | |||
|  |                 try { | |||
|  |                     leftParams = this.ControlParameters; | |||
|  |                     rightParams = otherImplementation.ControlParameters; | |||
|  |                 } catch (InvalidOperationException) { | |||
|  |                     return false; | |||
|  |                 } | |||
|  | 
 | |||
|  |                 Debug.Assert(leftParams != null, "leftParams shouldn't be null"); | |||
|  |                 Debug.Assert(rightParams != null, "rightParams shouldn't be null"); | |||
|  |                 if (leftParams.Count != rightParams.Count) { | |||
|  |                     return false; | |||
|  |                 } else { | |||
|  |                     return leftParams.OrderBy(p => p.Key).SequenceEqual(rightParams.OrderBy(p => p.Key)); | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  | 
 | |||
|  |             /// <summary> | |||
|  |             /// Validates the input control parameters and throws InvalidOperationException if they are not correct. | |||
|  |             /// </summary> | |||
|  |             /// <returns> | |||
|  |             /// Dictionary of control parameters. | |||
|  |             /// </returns> | |||
|  |             private IDictionary<string, object> BuildControlParametersDictionary() { | |||
|  |                 IDictionary<string, object> controlParameters = new Dictionary<string, object>(); | |||
|  | 
 | |||
|  |                 object[] inputControlParameters = this._inputControlParameters; | |||
|  | 
 | |||
|  |                 if (inputControlParameters == null || inputControlParameters.Length == 0) { | |||
|  |                     return controlParameters; | |||
|  |                 } | |||
|  |                 if (inputControlParameters.Length % 2 != 0) { | |||
|  |                     throw new InvalidOperationException(String.Format(CultureInfo.CurrentCulture, DataAnnotationsResources.UIHintImplementation_NeedEvenNumberOfControlParameters)); | |||
|  |                 } | |||
|  | 
 | |||
|  |                 for (int i = 0; i < inputControlParameters.Length; i += 2) { | |||
|  |                     object key = inputControlParameters[i]; | |||
|  |                     object value = inputControlParameters[i + 1]; | |||
|  |                     if (key == null) { | |||
|  |                         throw new InvalidOperationException( | |||
|  |                             String.Format( | |||
|  |                             CultureInfo.CurrentCulture, | |||
|  |                             DataAnnotationsResources.UIHintImplementation_ControlParameterKeyIsNull, | |||
|  |                             i)); | |||
|  |                     } | |||
|  | 
 | |||
|  |                     string keyString = key as string; | |||
|  |                     if (keyString == null) { | |||
|  |                         throw new InvalidOperationException( | |||
|  |                             String.Format( | |||
|  |                             CultureInfo.CurrentCulture, | |||
|  |                             DataAnnotationsResources.UIHintImplementation_ControlParameterKeyIsNotAString, | |||
|  |                             i, | |||
|  |                             inputControlParameters[i].ToString())); | |||
|  |                     } | |||
|  | 
 | |||
|  |                     if (controlParameters.ContainsKey(keyString)) { | |||
|  |                         throw new InvalidOperationException( | |||
|  |                             String.Format( | |||
|  |                             CultureInfo.CurrentCulture, | |||
|  |                             DataAnnotationsResources.UIHintImplementation_ControlParameterKeyOccursMoreThanOnce, | |||
|  |                             i, | |||
|  |                             keyString)); | |||
|  |                     } | |||
|  | 
 | |||
|  |                     controlParameters[keyString] = value; | |||
|  |                 } | |||
|  | 
 | |||
|  |                 return controlParameters; | |||
|  |             } | |||
|  |         } | |||
|  |     } | |||
|  | } |