using System.ComponentModel; using System.ComponentModel.DataAnnotations; using System.Data.Spatial; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Globalization; using System.Linq; using System.Reflection; using System.Web.DynamicData.ModelProviders; using System.Web.DynamicData.Util; namespace System.Web.DynamicData { /// /// Object that represents a database column used by dynamic data /// public class MetaColumn : IFieldFormattingOptions, IMetaColumn { private TypeCode _typeCode = TypeCode.Empty; private Type _type; private object _metadataCacheLock = new object(); // text, ntext, varchar(max), nvarchar(max) all have different maximum lengths so this is a minimum value // that ensures that all of the abovementioned columns get treated as long strings. private static readonly int s_longStringLengthCutoff = ((Int32.MaxValue >> 1) - 4); // Metadata related members private IMetaColumnMetadata _metadata; private bool? _scaffoldValueManual; private bool? _scaffoldValueDefault; public MetaColumn(MetaTable table, ColumnProvider columnProvider) { Table = table; Provider = columnProvider; } /// /// The collection of metadata attributes that apply to this column /// public AttributeCollection Attributes { get { return Metadata.Attributes; } } /// /// The CLR type of the property/column /// public Type ColumnType { get { if (_type == null) { // If it's an Nullable, work with T instead _type = Misc.RemoveNullableFromType(Provider.ColumnType); } return _type; } } /// /// The DataTypeAttribute used for the column /// public DataTypeAttribute DataTypeAttribute { get { return Metadata.DataTypeAttribute; } } /// /// This column's defalut value. It is typically used to populate the field when creating a new entry. /// public object DefaultValue { get { return Metadata.DefaultValue; } } /// /// A description for this column /// public virtual string Description { get { // return Metadata.Description; } } /// /// A friendly display name for this column /// public virtual string DisplayName { get { // Default to the Name if there is no DisplayName return Metadata.DisplayName ?? Name; } } /// /// The PropertyInfo of the property that represents this column on the entity type /// public PropertyInfo EntityTypeProperty { get { return Provider.EntityTypeProperty; } } /// /// The FilterUIHint used for the column /// public string FilterUIHint { get { return Metadata.FilterUIHint; } } /// /// Does this column contain binary data /// public bool IsBinaryData { get { return ColumnType == typeof(byte[]); } } /// /// meant to indicate that a member is an extra property that was declared in a partial class /// public bool IsCustomProperty { get { return Provider.IsCustomProperty; } } /// /// Is this column a floating point type (float, double, decimal) /// public bool IsFloatingPoint { get { return ColumnType == typeof(float) || ColumnType == typeof(double) || ColumnType == typeof(decimal); } } /// /// 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 bool IsForeignKeyComponent { get { return Provider.IsForeignKeyComponent; } } /// /// Is this column's value auto-generated in the database /// public bool IsGenerated { get { return Provider.IsGenerated; } } /// /// Is this column a integer /// public bool IsInteger { get { return ColumnType == typeof(byte) || ColumnType == typeof(short) || ColumnType == typeof(int) || ColumnType == typeof(long); } } /// /// Is this column a 'long' string. This is used to determine whether a textbox or textarea should be used. /// public bool IsLongString { get { return IsString && Provider.MaxLength >= s_longStringLengthCutoff; } } /// /// Is this column part if the table's primary key /// public bool IsPrimaryKey { get { return Provider.IsPrimaryKey; } } /// /// Is this a readonly column /// [SuppressMessage("Microsoft.Security", "CA2119:SealMethodsThatSatisfyPrivateInterfaces", Justification = "Interface denotes existence of property, not used for security.")] public virtual bool IsReadOnly { get { return Provider.IsReadOnly || Metadata.IsReadOnly || (Metadata.EditableAttribute != null && !Metadata.EditableAttribute.AllowEdit); } } /// /// Specifies if a read-only column (see IsReadOnly) allows for a value to be set on insert. /// The default value is false when the column is read-only; true when the column is not read-only. /// The default value can be override by using EditableAttribute (note that this will indicate that /// the column is meant to be read only). /// public bool AllowInitialValue { get { if (IsGenerated) { // always return false for generated columns, since that is a stronger statement. return false; } return Metadata.EditableAttribute.GetPropertyValue(a => a.AllowInitialValue, !IsReadOnly); } } /// /// Does this column require a value /// public bool IsRequired { get { return Metadata.RequiredAttribute != null; } } /// /// Is this column a string /// public bool IsString { get { return ColumnType == typeof(string); } } /// /// The maximun length allowed for this column (applies to string columns) /// public int MaxLength { get { var stringLengthAttribute = Metadata.StringLengthAttribute; return stringLengthAttribute != null ? stringLengthAttribute.MaximumLength : Provider.MaxLength; } } /// /// The MetaModel that this column belongs to /// public MetaModel Model { get { return Table.Model; } } /// /// The column's name /// public string Name { get { return Provider.Name; } } /// /// A value that can be used as a watermark in UI bound to value represented by this column. /// public virtual string Prompt { get { return Metadata.Prompt; } } /// /// the abstraction provider object that was used to construct this metacolumn. /// public ColumnProvider Provider { get; private set; } /// /// The error message used if this column is required and it is set to empty /// public string RequiredErrorMessage { get { var requiredAttribute = Metadata.RequiredAttribute; return requiredAttribute != null ? StringLocalizerUtil.GetLocalizedString(requiredAttribute, DisplayName) : String.Empty; } } /// /// Is it a column that should be displayed (e.g. in a GridView's auto generate mode) /// [SuppressMessage("Microsoft.Security", "CA2119:SealMethodsThatSatisfyPrivateInterfaces", Justification = "Interface denotes existence of property, not used for security.")] public virtual bool Scaffold { get { // If the value was explicitely set, that always takes precedence if (_scaffoldValueManual != null) { return _scaffoldValueManual.Value; } // If there is a DisplayAttribute with an explicit value, always honor it var displayAttribute = Metadata.DisplayAttribute; if (displayAttribute != null && displayAttribute.GetAutoGenerateField().HasValue) { return displayAttribute.GetAutoGenerateField().Value; } // If there is an explicit Scaffold attribute, always honor it var scaffoldAttribute = Metadata.ScaffoldColumnAttribute; if (scaffoldAttribute != null) { return scaffoldAttribute.Scaffold; } if (_scaffoldValueDefault == null) { _scaffoldValueDefault = ScaffoldNoCache; } return _scaffoldValueDefault.Value; } set { _scaffoldValueManual = value; } } /// /// Look at various pieces of data on the column to determine whether it's /// Scaffold mode should be on. This only gets called once per column and the result /// is cached /// internal virtual bool ScaffoldNoCache { get { // Any field with a UIHint should be included if (!String.IsNullOrEmpty(UIHint)) return true; // Skip columns that are part of a foreign key, since they are already 'covered' in the // strongly typed foreign key column if (IsForeignKeyComponent) return false; // Skip generated columns, which are not typically interesting if (IsGenerated) return false; // Always include non-generated primary keys if (IsPrimaryKey) return true; // Skip custom properties if (IsCustomProperty) return false; // Include strings and characters if (IsString) return true; if (ColumnType == typeof(char)) return true; // Include numbers if (IsInteger) return true; if (IsFloatingPoint) return true; // Include date related columns if (ColumnType == typeof(DateTime)) return true; if (ColumnType == typeof(TimeSpan)) return true; if (ColumnType == typeof(DateTimeOffset)) return true; // Include bools if (ColumnType == typeof(bool)) return true; // Include enums Type enumType; if (this.IsEnumType(out enumType)) return true; //Include spatial types if (ColumnType == typeof(DbGeography)) return true; if (ColumnType == typeof(DbGeometry)) return true; return false; } } /// /// A friendly short display name for this column. Meant to be used in GridView and similar controls where there might be /// limited column header space /// public virtual string ShortDisplayName { get { // Default to the DisplayName if there is no ShortDisplayName return Metadata.ShortDisplayName ?? DisplayName; } } /// /// The expression used to determine the sort order for this column /// public string SortExpression { get { return SortExpressionInternal; } } internal virtual string SortExpressionInternal { get { return Provider.IsSortable ? Name : string.Empty; } } /// /// The MetaTable that this column belongs to /// public MetaTable Table { get; private set; } /// /// The TypeCode of this column. It is derived from the ColumnType /// public TypeCode TypeCode { get { if (_typeCode == TypeCode.Empty) { _typeCode = DataSourceUtil.TypeCodeFromType(ColumnType); } return _typeCode; } } /// /// The UIHint used for the column /// [SuppressMessage("Microsoft.Security", "CA2119:SealMethodsThatSatisfyPrivateInterfaces", Justification = "Interface denotes existence of property, not used for security.")] public virtual string UIHint { get { return Metadata.UIHint; } } #region IFieldFormattingOptions Members /// /// Same semantic as the same property on System.Web.UI.WebControls.BoundField /// public bool ApplyFormatInEditMode { get { return Metadata.ApplyFormatInEditMode; } } /// /// Same semantic as the same property on System.Web.UI.WebControls.BoundField /// public bool ConvertEmptyStringToNull { get { return Metadata.ConvertEmptyStringToNull; } } /// /// Same semantic as the same property on System.Web.UI.WebControls.BoundField /// public string DataFormatString { get { return Metadata.DataFormatString; } } /// /// Same semantic as the same property on System.Web.UI.WebControls.BoundField /// public bool HtmlEncode { get { return Metadata.HtmlEncode; } } /// /// Same semantic as the same property on System.Web.UI.WebControls.BoundField /// public string NullDisplayText { get { return Metadata.NullDisplayText; } } #endregion /// /// Build the attribute collection, later made available through the Attributes property /// protected virtual AttributeCollection BuildAttributeCollection() { return Provider.Attributes; } /// /// Perform initialization logic for this column /// internal protected virtual void Initialize() { } /// /// Resets cached column metadata (i.e. information coming from attributes). The metadata cache will be rebuilt /// the next time any metadata-derived information gets requested. /// public void ResetMetadata() { _metadata = null; } /// /// Shows the column name. Mostly for debugging purpose. /// [SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase")] public override string ToString() { return GetType().Name + " " + Name; } internal IMetaColumnMetadata Metadata { get { // Use a local to avoid returning null if ResetMetadata is called IMetaColumnMetadata metadata = _metadata; if (metadata == null) { metadata = new MetaColumnMetadata(this); _metadata = metadata; } return metadata; } set { // settable for unit testing _metadata = value; } } #region Metadata abstraction internal interface IMetaColumnMetadata { AttributeCollection Attributes { get; } DisplayAttribute DisplayAttribute { get; } bool ApplyFormatInEditMode { get; } bool ConvertEmptyStringToNull { get; } bool HtmlEncode { get; } string DataFormatString { get; } DataTypeAttribute DataTypeAttribute { get; } object DefaultValue { get; } string Description { get; } string DisplayName { get; } string FilterUIHint { get; } string ShortDisplayName { get; } string NullDisplayText { get; } string Prompt { get; } RequiredAttribute RequiredAttribute { get; } ScaffoldColumnAttribute ScaffoldColumnAttribute { get; } StringLengthAttribute StringLengthAttribute { get; } string UIHint { get; } bool IsReadOnly { get; } EditableAttribute EditableAttribute { get; } } internal class MetaColumnMetadata : IMetaColumnMetadata { private MetaColumn Column { get; set; } public AttributeCollection Attributes { get; private set; } public MetaColumnMetadata(MetaColumn column) { Debug.Assert(column != null); Column = column; Attributes = Column.BuildAttributeCollection(); DisplayAttribute = Attributes.FirstOrDefault(); DataTypeAttribute = Attributes.FirstOrDefault() ?? GetDefaultDataTypeAttribute(); DescriptionAttribute = Attributes.FirstOrDefault(); DefaultValueAttribute = Attributes.FirstOrDefault(); DisplayNameAttribute = Attributes.FirstOrDefault(); RequiredAttribute = Attributes.FirstOrDefault(); ScaffoldColumnAttribute = Attributes.FirstOrDefault(); StringLengthAttribute = Attributes.FirstOrDefault(); UIHint = GetHint(a => a.PresentationLayer, a => a.UIHint); FilterUIHint = GetHint(a => a.PresentationLayer, a => a.FilterUIHint); EditableAttribute = Attributes.FirstOrDefault(); IsReadOnly = Attributes.GetAttributePropertyValue(a => a.IsReadOnly, false); var displayFormatAttribute = Attributes.FirstOrDefault() ?? (DataTypeAttribute != null ? DataTypeAttribute.DisplayFormat : null); ApplyFormatInEditMode = displayFormatAttribute.GetPropertyValue(a => a.ApplyFormatInEditMode, false); ConvertEmptyStringToNull = displayFormatAttribute.GetPropertyValue(a => a.ConvertEmptyStringToNull, true); DataFormatString = displayFormatAttribute.GetPropertyValue(a => a.DataFormatString, String.Empty); NullDisplayText = displayFormatAttribute.GetPropertyValue(a => a.NullDisplayText, String.Empty); HtmlEncode = displayFormatAttribute.GetPropertyValue(a => a.HtmlEncode, true); } public DisplayAttribute DisplayAttribute { get; private set; } public bool ApplyFormatInEditMode { get; private set; } public bool ConvertEmptyStringToNull { get; private set; } public string DataFormatString { get; private set; } public DataTypeAttribute DataTypeAttribute { get; private set; } public object DefaultValue { get { return DefaultValueAttribute.GetPropertyValue(a => a.Value, null); } } private DefaultValueAttribute DefaultValueAttribute { get; set; } public string Description { get { return DisplayAttribute.GetLocalizedDescription() ?? DescriptionAttribute.GetPropertyValue(a => a.Description, null); } } private DescriptionAttribute DescriptionAttribute { get; set; } public string DisplayName { get { return DisplayAttribute.GetLocalizedName() ?? DisplayNameAttribute.GetPropertyValue(a => a.DisplayName, null); } } public string ShortDisplayName { get { return DisplayAttribute.GetLocalizedShortName(); } } private DisplayNameAttribute DisplayNameAttribute { get; set; } public string FilterUIHint { get; private set; } public EditableAttribute EditableAttribute { get; private set; } public bool IsReadOnly { get; private set; } public string NullDisplayText { get; private set; } public string Prompt { get { return DisplayAttribute.GetLocalizedPrompt(); } } public RequiredAttribute RequiredAttribute { get; private set; } public ScaffoldColumnAttribute ScaffoldColumnAttribute { get; private set; } public StringLengthAttribute StringLengthAttribute { get; private set; } public string UIHint { get; private set; } private DataTypeAttribute GetDefaultDataTypeAttribute() { if (Column.IsString) { if (Column.IsLongString) { return new DataTypeAttribute(DataType.MultilineText); } else { return new DataTypeAttribute(DataType.Text); } } return null; } private string GetHint(Func presentationLayerPropertyAccessor, Func hintPropertyAccessor) where T : Attribute { var uiHints = Attributes.OfType(); var presentationLayerNotSpecified = uiHints.Where(a => String.IsNullOrEmpty(presentationLayerPropertyAccessor(a))); var presentationLayerSpecified = uiHints.Where(a => !String.IsNullOrEmpty(presentationLayerPropertyAccessor(a))); T uiHintAttribute = presentationLayerSpecified.FirstOrDefault(a => presentationLayerPropertyAccessor(a).ToLower(CultureInfo.InvariantCulture) == "webforms" || presentationLayerPropertyAccessor(a).ToLower(CultureInfo.InvariantCulture) == "mvc") ?? presentationLayerNotSpecified.FirstOrDefault(); return uiHintAttribute.GetPropertyValue(hintPropertyAccessor); } public bool HtmlEncode { get; set; } } #endregion string IMetaColumn.Description { get { return Description; } } string IMetaColumn.DisplayName { get { return DisplayName; } } string IMetaColumn.Prompt { get { return Prompt; } } string IMetaColumn.ShortDisplayName { get { return ShortDisplayName; } } IMetaTable IMetaColumn.Table { get { return Table; } } IMetaModel IMetaColumn.Model { get { return Model; } } } }