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 { /// /// Attribute to provide a hint to the presentation layer about what control it should use /// [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; /// /// Gets the name of the control that is most appropriate for this associated property or field /// public string UIHint { get { return this._implementation.UIHint; } } /// /// Gets the name of the presentation layer that supports the control type in /// public string PresentationLayer { get { return this._implementation.PresentationLayer; } } /// /// Gets the name-value pairs used as parameters to the control's constructor /// /// is thrown if the current attribute is ill-formed. public IDictionary ControlParameters { get { return this._implementation.ControlParameters; } } #if !SILVERLIGHT /// /// Gets a unique identifier for this attribute. /// public override object TypeId { get { return this; } } #endif /// /// Constructor that accepts the name of the control, without specifying which presentation layer to use /// /// The name of the UI control. public UIHintAttribute(string uiHint) : this(uiHint, null, new object[0]) { } /// /// Constructor that accepts both the name of the control as well as the presentation layer /// /// The name of the control to use /// The name of the presentation layer that supports this control public UIHintAttribute(string uiHint, string presentationLayer) : this(uiHint, presentationLayer, new object[0]) { } /// /// Full constructor that accepts the name of the control, presentation layer, and optional parameters /// to use when constructing the control /// /// The name of the control /// The presentation layer /// The list of parameters for the control 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 _controlParameters; private object[] _inputControlParameters; /// /// Gets the name of the control that is most appropriate for this associated property or field /// public string UIHint { get; private set; } /// /// Gets the name of the presentation layer that supports the control type in /// public string PresentationLayer { get; private set; } public IDictionary 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); } } /// /// Returns the hash code for this UIHintAttribute. /// /// A 32-bit signed integer hash code. public override int GetHashCode() { var a = this.UIHint ?? String.Empty; var b = this.PresentationLayer ?? String.Empty; return a.GetHashCode() ^ b.GetHashCode(); } /// /// Determines whether this instance of UIHintAttribute and a specified object, /// which must also be a UIHintAttribute object, have the same value. /// /// An System.Object. /// true if obj is a UIHintAttribute and its value is the same as this instance; otherwise, false. 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 leftParams; IDictionary 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)); } } /// /// Validates the input control parameters and throws InvalidOperationException if they are not correct. /// /// /// Dictionary of control parameters. /// private IDictionary BuildControlParametersDictionary() { IDictionary controlParameters = new Dictionary(); 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; } } } }