using System.Collections.Generic; using System.ComponentModel; using System.ComponentModel.DataAnnotations; using System.Diagnostics.CodeAnalysis; using System.Reflection; using System.Web.DynamicData; using System.Web.DynamicData.Util; namespace System.Web.DynamicData.ModelProviders { /// /// Base provider class for columns. /// Each provider type (e.g. Linq To Sql, Entity Framework, 3rd party) extends this class. /// public abstract class ColumnProvider { private bool? _isReadOnly; /// /// ctor /// /// the table this column belongs to protected ColumnProvider(TableProvider table) { if (table == null) { throw new ArgumentNullException("table"); } Table = table; } /// /// readable representation /// /// [SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase")] public override string ToString() { // To help identifying objects in debugger return Name ?? base.ToString(); } internal virtual PropertyDescriptor PropertyDescriptor { get { return Table.GetTypeDescriptor().GetProperties().Find(Name, true/*ignoreCase*/); } } public virtual AttributeCollection Attributes { get { var propertyDescriptor = PropertyDescriptor; var attributes = propertyDescriptor != null ? propertyDescriptor.Attributes : AttributeCollection.Empty; return AddDefaultAttributes(this, attributes); } } protected static AttributeCollection AddDefaultAttributes(ColumnProvider columnProvider, AttributeCollection attributes) { List extraAttributes = new List(); // If there is no required attribute and the Provider says required, add one var requiredAttribute = attributes.FirstOrDefault(); if (requiredAttribute == null && !columnProvider.Nullable) { extraAttributes.Add(new RequiredAttribute()); } // If there is no StringLength attribute and it's a string, add one var stringLengthAttribute = attributes.FirstOrDefault(); int maxLength = columnProvider.MaxLength; if (stringLengthAttribute == null && columnProvider.ColumnType == typeof(String) && maxLength > 0) { extraAttributes.Add(new StringLengthAttribute(maxLength)); } // If we need any extra attributes, create a new collection if (extraAttributes.Count > 0) { attributes = AttributeCollection.FromExisting(attributes, extraAttributes.ToArray()); } return attributes; } /// /// The name of the column /// public virtual string Name { get; protected set; } /// /// The CLR type of the column /// public virtual Type ColumnType { get; protected set; } /// /// Is this column a primary key in its table /// public virtual bool IsPrimaryKey { get; protected set; } /// /// Specifies if this column is read only /// public virtual bool IsReadOnly { get { if (_isReadOnly == null) { var propertyDescriptor = PropertyDescriptor; _isReadOnly = propertyDescriptor != null ? propertyDescriptor.IsReadOnly : false; } return _isReadOnly.Value; } protected set { _isReadOnly = value; } } /// /// Is it a database generated column /// public virtual bool IsGenerated { get; protected set; } /// /// Returns whether the underlying model supports sorting of the table on this column /// [SuppressMessage("Microsoft.Naming", "CA1704:IdentifiersShouldBeSpelledCorrectly", MessageId = "Sortable", Justification="It's a valid word")] public virtual bool IsSortable { get; protected set; } /// /// The maximun length allowed for this column (applies to string columns) /// public virtual int MaxLength { get; protected set; } /// /// Does it allow null values (meaning it is not required) /// public virtual bool Nullable { get; protected set; } /// /// meant to indicate that a member is an extra property that was declared in a partial class /// public virtual bool IsCustomProperty { get; protected set; } /// /// If the column represents and association with anther table, this returns the association information. /// Otherwise, null is returned. /// public virtual AssociationProvider Association { get; protected set; } /// /// The table that this column belongs to /// public TableProvider Table { get; private set; } /// /// The PropertyInfo of the property that represents this column on the entity type /// public virtual PropertyInfo EntityTypeProperty { get; protected set; } /// /// This is set for columns that are part of a foreign key. Note that it is NOT set for /// the strongly typed entity ref columns (though those columns 'use' one or more columns /// where IsForeignKeyComponent is set). /// public virtual bool IsForeignKeyComponent { get; protected set; } } }