You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			356 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			356 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Collections;
 | |
| using System.Collections.Generic;
 | |
| using System.Collections.Specialized;
 | |
| using System.ComponentModel;
 | |
| using System.Data;
 | |
| using System.Diagnostics;
 | |
| using System.Globalization;
 | |
| using System.Linq;
 | |
| using System.Linq.Expressions;
 | |
| using System.Text;
 | |
| using System.Web.DynamicData.Util;
 | |
| using System.Web.Resources;
 | |
| using System.Web.Routing;
 | |
| using System.Web.UI;
 | |
| using System.Web.UI.WebControls;
 | |
| 
 | |
| namespace System.Web.DynamicData {
 | |
|     internal static class Misc {
 | |
|         public static HttpContextWrapper ToWrapper(this HttpContext context) {
 | |
|             return new HttpContextWrapper(context);
 | |
|         }
 | |
| 
 | |
|         public static object GetRealDataItem(object dataItem) {
 | |
|             if (dataItem is ICustomTypeDescriptor) {
 | |
|                 // Unwrap EF object
 | |
|                 dataItem = ((ICustomTypeDescriptor)dataItem).GetPropertyOwner(null);
 | |
|             }
 | |
|             return dataItem;
 | |
|         }
 | |
| 
 | |
|         // Walks the type hierachy up to endingType (assuming startingType is a subtype of starting type)
 | |
|         // trying to find a meta table.
 | |
|         public static MetaTable GetTableFromTypeHierarchy(Type entityType) {
 | |
|             if (entityType == null) {
 | |
|                 throw new ArgumentNullException("entityType");
 | |
|             }
 | |
| 
 | |
|             Type type = entityType;
 | |
|             while (type != null) {
 | |
|                 MetaTable table;
 | |
|                 if (MetaTable.TryGetTable(type, out table)) {
 | |
|                     return table;
 | |
|                 }
 | |
|                 type = type.BaseType;
 | |
|             }
 | |
| 
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         public static Type RemoveNullableFromType(Type type) {
 | |
|             return Nullable.GetUnderlyingType(type) ?? type;
 | |
|         }
 | |
| 
 | |
|         internal static bool IsColumnInDictionary(IMetaColumn column, IDictionary<string, object> values) {
 | |
|             if (column == null) {
 | |
|                 throw new ArgumentNullException("column");
 | |
|             }
 | |
|             if (values == null) {
 | |
|                 throw new ArgumentNullException("values");
 | |
|             }
 | |
|             IMetaForeignKeyColumn foreignKeyColumn = column as IMetaForeignKeyColumn;
 | |
|             if (foreignKeyColumn != null) {
 | |
|                 return foreignKeyColumn.ForeignKeyNames.All(fkName => values.ContainsKey(fkName));
 | |
|             }
 | |
|             return values.ContainsKey(column.Name);
 | |
|         }
 | |
| 
 | |
|         internal static IDictionary<string, object> ConvertObjectToDictionary(object instance) {
 | |
|             if (instance == null) {
 | |
|                 throw new ArgumentNullException("instance");
 | |
|             }
 | |
|             Dictionary<string, object> values = new Dictionary<string, object>();
 | |
|             var props = TypeDescriptor.GetProperties(instance);
 | |
|             foreach (PropertyDescriptor p in props) {
 | |
|                 values[p.Name] = p.GetValue(instance);
 | |
|             }
 | |
|             return values;
 | |
|         }
 | |
| 
 | |
|         public static T ChangeType<T>(object value) {
 | |
|             return (T)ChangeType(value, typeof(T));
 | |
|         }
 | |
| 
 | |
|         public static object ChangeType(object value, Type type) {
 | |
|             if (type == null) {
 | |
|                 throw new ArgumentNullException("type");
 | |
|             }
 | |
| 
 | |
|             if (value == null) {
 | |
|                 if (TypeAllowsNull(type)) {
 | |
|                     return null;
 | |
|                 }
 | |
|                 return Convert.ChangeType(value, type, CultureInfo.CurrentCulture);
 | |
|             }
 | |
| 
 | |
|             type = RemoveNullableFromType(type);
 | |
| 
 | |
|             if (value.GetType() == type) {
 | |
|                 return value;
 | |
|             }
 | |
| 
 | |
|             TypeConverter converter = TypeDescriptor.GetConverter(type);
 | |
|             if (converter.CanConvertFrom(value.GetType())) {
 | |
|                 return converter.ConvertFrom(value);
 | |
|             }
 | |
| 
 | |
|             TypeConverter otherConverter = TypeDescriptor.GetConverter(value.GetType());
 | |
|             if (otherConverter.CanConvertTo(type)) {
 | |
|                 return otherConverter.ConvertTo(value, type);
 | |
|             }
 | |
| 
 | |
|             throw new InvalidOperationException(String.Format(
 | |
|                             CultureInfo.CurrentCulture,
 | |
|                             DynamicDataResources.Misc_CannotConvertType, value.GetType(), type));
 | |
|         }
 | |
| 
 | |
|         internal static bool TypeAllowsNull(Type type) {
 | |
|             return Nullable.GetUnderlyingType(type) != null || !type.IsValueType;
 | |
|         }
 | |
| 
 | |
|         public static ContainerType FindContainerType(Control control) {
 | |
|             if (control == null) {
 | |
|                 throw new ArgumentNullException("control");
 | |
|             }
 | |
| 
 | |
|             Control container = control;
 | |
|             // Walk up NamingContainers until we find one of the DataBound control interfaces
 | |
|             while (container != null) {
 | |
|                 if (container is IDataBoundItemControl) {
 | |
|                     return ContainerType.Item;
 | |
|                 }
 | |
|                 else if (container is IDataBoundListControl || container is Repeater) {
 | |
|                     return ContainerType.List;
 | |
|                 }
 | |
|                 container = container.NamingContainer;
 | |
|             }
 | |
|             // Default container type is a list if none of the known 
 | |
|             // interfaces are found
 | |
|             return ContainerType.List;
 | |
|         }
 | |
| 
 | |
|         public static IOrderedDictionary GetEnumNamesAndValues(Type enumType) {
 | |
|             Debug.Assert(enumType != null);
 | |
|             Debug.Assert(enumType.IsEnum);
 | |
|             OrderedDictionary result = new OrderedDictionary();
 | |
|             var enumEntries = from e in Enum.GetValues(enumType).OfType<object>()
 | |
|                               select new EnumEntry {
 | |
|                                   // 
 | |
|                                   Name = Enum.GetName(enumType, e),
 | |
|                                   UnderlyingValue = GetUnderlyingTypeValue(enumType, e)
 | |
|                               };
 | |
|             foreach (var entry in enumEntries.OrderBy(e => e.UnderlyingValue)) {
 | |
|                 result.Add(entry.Name, entry.UnderlyingValue.ToString());
 | |
|             }
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         private struct EnumEntry {
 | |
|             public string Name { get; set; }
 | |
|             public object UnderlyingValue { get; set; }
 | |
|         }
 | |
| 
 | |
|         public static object GetUnderlyingTypeValue(Type enumType, object enumValue) {
 | |
|             return Convert.ChangeType(enumValue, Enum.GetUnderlyingType(enumType), CultureInfo.InvariantCulture);
 | |
|         }
 | |
| 
 | |
|         public static string GetUnderlyingTypeValueString(Type enumType, object enumValue) {
 | |
|             return GetUnderlyingTypeValue(enumType, enumValue).ToString();
 | |
|         }
 | |
| 
 | |
|         public static string PersistListToCommaSeparatedString(IList<object> list) {
 | |
|             // Special case empty and single lists
 | |
|             if (list == null || list.Count == 0)
 | |
|                 return String.Empty;
 | |
|             if (list.Count == 1) {
 | |
|                 return list[0] == null ? String.Empty : list[0].ToString().TrimEnd();
 | |
|             }
 | |
| 
 | |
|             var builder = new StringBuilder();
 | |
|             bool first = true;
 | |
|             bool hasNonNullItem = false;
 | |
|             foreach (object o in list) {
 | |
|                 if (!first) {
 | |
|                     builder.Append(",");
 | |
|                 }
 | |
| 
 | |
|                 if (o != null) {
 | |
|                     // 
 | |
|                     builder.Append(o.ToString().TrimEnd());
 | |
| 
 | |
|                     hasNonNullItem = true;
 | |
|                 }
 | |
|                 first = false;
 | |
|             }
 | |
| 
 | |
|             // If all the parts are null, return empty string instead of the comma separated list
 | |
|             if (!hasNonNullItem)
 | |
|                 return String.Empty;
 | |
| 
 | |
|             return builder.ToString();
 | |
|         }
 | |
| 
 | |
|         // 
 | |
|         public static object[] GetKeyValues(IList<MetaColumn> keyMembers, object entity) {
 | |
|             object[] values = new object[keyMembers.Count];
 | |
| 
 | |
|             int index = 0;
 | |
|             foreach (MetaColumn pkMember in keyMembers) {
 | |
|                 values[index++] = DataBinder.GetPropertyValue(entity, pkMember.Name);
 | |
|             }
 | |
| 
 | |
|             return values;
 | |
|         }
 | |
| 
 | |
|         public static string[] ParseCommaSeparatedString(string stringList) {
 | |
|             // 
 | |
|             return stringList.Split(',');
 | |
|         }
 | |
| 
 | |
|         public static IQueryable BuildSortQueryable(IQueryable query, IMetaTable table) {
 | |
|             IMetaColumn sortColumn = table.SortColumn;
 | |
|             if (sortColumn.IsCustomProperty) {
 | |
|                 // An extra property can't be optimized on server
 | |
|                 // 
 | |
|                 var data = query.OfType<object>().AsEnumerable();
 | |
|                 Func<object, object> lambda = row => DataBinder.GetPropertyValue(row, sortColumn.Name);
 | |
|                 if (table.SortDescending) {
 | |
|                     query = data.OrderByDescending<object, object>(lambda).AsQueryable();
 | |
|                 }
 | |
|                 else {
 | |
|                     query = data.OrderBy<object, object>(lambda).AsQueryable();
 | |
|                 }
 | |
|             }
 | |
|             else {
 | |
|                 // Build custom expression to optimize sorting on server
 | |
|                 // 
 | |
|                 var parameter = Expression.Parameter(query.ElementType, "row");
 | |
|                 LambdaExpression lambda = null;
 | |
|                 IMetaForeignKeyColumn foreignKeyColumn = sortColumn as IMetaForeignKeyColumn;
 | |
|                 if (foreignKeyColumn != null) {
 | |
|                     // e.g. product => product.Category.CategoryName
 | |
|                     var foreignKeySortColumn = foreignKeyColumn.ParentTable.SortColumn;
 | |
|                     lambda = Expression.Lambda(Expression.Property(Expression.Property(parameter, sortColumn.Name), foreignKeySortColumn.Name), parameter);
 | |
|                 }
 | |
|                 else {
 | |
|                     // e.g. product => product.ProductName
 | |
|                     lambda = Expression.Lambda(Expression.Property(parameter, sortColumn.Name), parameter);
 | |
|                 }
 | |
|                 string ordering = table.SortDescending ? "OrderByDescending" : "OrderBy";
 | |
|                 var expression = Expression.Call(typeof(Queryable), ordering, new Type[] { query.ElementType, lambda.Body.Type }, query.Expression, lambda);
 | |
|                 query = query.Provider.CreateQuery(expression);
 | |
|             }
 | |
|             return query;
 | |
|         }
 | |
| 
 | |
|         // Fill a ListItemCollection with all the entries from a table
 | |
|         public static void FillListItemCollection(IMetaTable table, ListItemCollection listItemCollection) {
 | |
|             foreach (var o in table.GetQuery()) {
 | |
|                 string text = table.GetDisplayString(o);
 | |
|                 string value = table.GetPrimaryKeyString(o);
 | |
|                 listItemCollection.Add(new ListItem(text, value.TrimEnd()));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static void ExtractValuesFromBindableControls(IOrderedDictionary dictionary, Control container) {
 | |
|             IBindableControl bindableControl = container as IBindableControl;
 | |
|             if (bindableControl != null) {
 | |
|                 bindableControl.ExtractValues(dictionary);
 | |
|             }
 | |
|             foreach (Control childControl in container.Controls) {
 | |
|                 ExtractValuesFromBindableControls(dictionary, childControl);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         /// Walks up the stack of NamingContainers starting at 'control' to find a control with the ID 'controlID'.
 | |
|         /// Copied from DataBoundControlHelper.FindControl (System.Web)
 | |
|         /// </devdoc>
 | |
|         public static Control FindControl(Control control, string controlID) {
 | |
|             Debug.Assert(control != null, "control should not be null");
 | |
|             //Debug.Assert(!String.IsNullOrEmpty(controlID), "controlID should not be empty");
 | |
|             Control currentContainer = control;
 | |
|             Control foundControl = null;
 | |
| 
 | |
|             if (control == control.Page) {
 | |
|                 // If we get to the Page itself while we're walking up the
 | |
|                 // hierarchy, just return whatever item we find (if anything)
 | |
|                 // since we can't walk any higher.
 | |
|                 return control.FindControl(controlID);
 | |
|             }
 | |
| 
 | |
|             while (foundControl == null && currentContainer != control.Page) {
 | |
|                 currentContainer = currentContainer.NamingContainer;
 | |
|                 if (currentContainer == null) {
 | |
|                     throw new HttpException(String.Format(CultureInfo.CurrentCulture,
 | |
|                             DynamicDataResources.Misc_NoNamingContainer,
 | |
|                             control.GetType().Name, control.ID));
 | |
|                 }
 | |
|                 foundControl = currentContainer.FindControl(controlID);
 | |
|             }
 | |
| 
 | |
|             return foundControl;
 | |
|         }
 | |
| 
 | |
|         public static string GetRouteValue(string key) {
 | |
|             RequestContext requestContext = DynamicDataRouteHandler.GetRequestContext(HttpContext.Current);
 | |
|             object value;
 | |
|             if (!requestContext.RouteData.Values.TryGetValue(key, out value)) {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             return value as string;
 | |
|         }
 | |
| 
 | |
|         public static string SanitizeQueryStringValue(object value) {
 | |
|             if (value == null)
 | |
|                 return null;
 | |
| 
 | |
|             string strValue = value.ToString();
 | |
| 
 | |
|             // Trim trailing spaces, as they are typically meaningless, and make the url look ugly
 | |
|             return strValue.TrimEnd();
 | |
|         }
 | |
| 
 | |
|         internal static long CombineHashCodes(object o1, object o2) {
 | |
|             // Start with a seed (obtained from String.GetHashCode implementation)
 | |
|             long combinedHash = 5381;
 | |
| 
 | |
|             combinedHash = AddHashCode(combinedHash, o1);
 | |
|             combinedHash = AddHashCode(combinedHash, o2);
 | |
| 
 | |
|             return combinedHash;
 | |
|         }
 | |
| 
 | |
|         // Return a single hash code for 3 objects
 | |
|         internal static long CombineHashCodes(object o1, object o2, object o3) {
 | |
|             // Start with a seed (obtained from String.GetHashCode implementation)
 | |
|             long combinedHash = 5381;
 | |
| 
 | |
|             combinedHash = AddHashCode(combinedHash, o1);
 | |
|             combinedHash = AddHashCode(combinedHash, o2);
 | |
|             combinedHash = AddHashCode(combinedHash, o3);
 | |
| 
 | |
|             return combinedHash;
 | |
|         }
 | |
| 
 | |
|         private static long AddHashCode(long currentHash, object o) {
 | |
|             if (o == null)
 | |
|                 return currentHash;
 | |
| 
 | |
|             return ((currentHash << 5) + currentHash) ^ o.GetHashCode();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 |