You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			199 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			199 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="PropertyMapper.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | /* | ||
|  |  * PropertyMapper.cs | ||
|  |  * | ||
|  |  * Copyright (c) 1999 Microsoft Corporation | ||
|  |  */ | ||
|  | 
 | ||
|  | namespace System.Web.UI { | ||
|  |     using System; | ||
|  |     using System.Collections; | ||
|  |     using System.Reflection; | ||
|  |     using System.Web.Util; | ||
|  |     using System.ComponentModel; | ||
|  | 
 | ||
|  |     internal sealed class PropertyMapper { | ||
|  |         private const char PERSIST_CHAR = '-'; | ||
|  |         private const char OM_CHAR = '.'; | ||
|  |         private const string STR_OM_CHAR = "."; | ||
|  | 
 | ||
|  | 
 | ||
|  |         /* | ||
|  |          * Maps persisted attribute names to the object model equivalents. | ||
|  |          * This class should not be instantiated by itself. | ||
|  |          */ | ||
|  |         private PropertyMapper() { | ||
|  |         } | ||
|  | 
 | ||
|  |         /* | ||
|  |          * Returns the PropertyInfo or FieldInfo corresponding to the | ||
|  |          * specified property name. | ||
|  |          */ | ||
|  |         internal static MemberInfo GetMemberInfo(Type ctrlType, string name, out string nameForCodeGen) { | ||
|  | 
 | ||
|  |             Type currentType = ctrlType; | ||
|  |             PropertyInfo propInfo = null; | ||
|  |             FieldInfo fieldInfo = null; | ||
|  | 
 | ||
|  |             string mappedName = MapNameToPropertyName(name); | ||
|  |             nameForCodeGen = null; | ||
|  | 
 | ||
|  |             int startIndex = 0; | ||
|  |             while (startIndex < mappedName.Length) {   // parse thru dots of object model to locate PropertyInfo | ||
|  |                 string propName; | ||
|  |                 int index = mappedName.IndexOf(OM_CHAR, startIndex); | ||
|  | 
 | ||
|  |                 if (index < 0) { | ||
|  |                     propName = mappedName.Substring(startIndex); | ||
|  |                     startIndex = mappedName.Length; | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     propName = mappedName.Substring(startIndex, index - startIndex); | ||
|  |                     startIndex = index + 1; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 BindingFlags flags = BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static | BindingFlags.IgnoreCase; | ||
|  | 
 | ||
|  |                 //  | ||
|  | 
 | ||
|  | 
 | ||
|  | 
 | ||
|  |                 try { | ||
|  |                     propInfo = TargetFrameworkUtil.GetProperty(currentType, propName, flags); | ||
|  |                 } | ||
|  |                 catch (AmbiguousMatchException) { | ||
|  |                     flags |= BindingFlags.DeclaredOnly; | ||
|  |                     propInfo = TargetFrameworkUtil.GetProperty(currentType, propName, flags); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (propInfo == null) {   // could not find a public property, look for a public field | ||
|  |                     fieldInfo = TargetFrameworkUtil.GetField(currentType, propName, flags); | ||
|  |                     if (fieldInfo == null) { | ||
|  |                         nameForCodeGen = null; | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 propName = null; | ||
|  |                 if (propInfo != null) {   // found a public property | ||
|  |                     currentType = propInfo.PropertyType; | ||
|  |                     propName = propInfo.Name; | ||
|  |                 } | ||
|  |                 else {   // found a public field | ||
|  |                     currentType = fieldInfo.FieldType; | ||
|  |                     propName = fieldInfo.Name; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // Throw if the type of not CLS-compliant (ASURT 83438) | ||
|  |                 if (!IsTypeCLSCompliant(currentType)) { | ||
|  |                     throw new HttpException(SR.GetString(SR.Property_Not_ClsCompliant, name, ctrlType.FullName, currentType.FullName)); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (propName != null) { | ||
|  |                     if (nameForCodeGen == null) | ||
|  |                         nameForCodeGen = propName; | ||
|  |                     else | ||
|  |                         nameForCodeGen += STR_OM_CHAR + propName; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if (propInfo != null) | ||
|  |                 return propInfo; | ||
|  |             else | ||
|  |                 return fieldInfo; | ||
|  |         } | ||
|  | 
 | ||
|  |         private static bool IsTypeCLSCompliant(Type type) { | ||
|  | 
 | ||
|  |             // We used to check this by looking up the CLSCompliantAttribute, but that was | ||
|  |             // way too inefficent.  So instead, we just juck for a few specific types | ||
|  |             if ((type == typeof(SByte)) || | ||
|  |                 (type == typeof(TypedReference)) || | ||
|  |                 (type == typeof(UInt16)) || | ||
|  |                 (type == typeof(UInt32)) || | ||
|  |                 (type == typeof(UInt64)) || | ||
|  |                 (type == typeof(UIntPtr))) { | ||
|  |                 return false; | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         /* | ||
|  |          * Maps the specified persisted name to its object model equivalent. | ||
|  |          * The convention is to map all dashes to dots. | ||
|  |          * For example :  Font-Size maps to Font.Size | ||
|  |          *                HeaderStyle-Font-Name maps to HeaderStyle.Font.Name | ||
|  |          */ | ||
|  |         internal static string MapNameToPropertyName(string attrName) { | ||
|  |             return attrName.Replace(PERSIST_CHAR,OM_CHAR); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static object LocatePropertyObject(object obj, string mappedName, out string propertyName, bool inDesigner) { | ||
|  |             object currentObject = obj; | ||
|  |             Type currentType = obj.GetType(); | ||
|  |             propertyName = null; | ||
|  |             int index; | ||
|  |             int startIndex = 0; | ||
|  | 
 | ||
|  |             // step through the dots of the object model to extract the PropertyInfo | ||
|  |             // and object on which the property will be set | ||
|  |             while (startIndex < mappedName.Length) { | ||
|  |                 index = mappedName.IndexOf(OM_CHAR, startIndex); | ||
|  | 
 | ||
|  |                 // If we didn't find a separator, we're on the last piece | ||
|  |                 if (index < 0) | ||
|  |                     break; | ||
|  | 
 | ||
|  |                 // There is a sub property, so get its info and iterate | ||
|  | 
 | ||
|  |                 propertyName = mappedName.Substring(startIndex, index - startIndex); | ||
|  |                 startIndex = index + 1; | ||
|  | 
 | ||
|  |                 currentObject = FastPropertyAccessor.GetProperty(currentObject, propertyName, inDesigner); | ||
|  | 
 | ||
|  |                 if (currentObject == null) | ||
|  |                     return null; | ||
|  |             } | ||
|  | 
 | ||
|  |             // Avoid a useless call to Substring if possible | ||
|  |             if (startIndex > 0) | ||
|  |                 propertyName = mappedName.Substring(startIndex); | ||
|  |             else | ||
|  |                 propertyName = mappedName; | ||
|  | 
 | ||
|  |             return currentObject; | ||
|  |         } | ||
|  | 
 | ||
|  |         /* | ||
|  |          * Walks the object model using the mapped property name to get the | ||
|  |          * value of an instance's property. | ||
|  |          */ | ||
|  |         internal static PropertyDescriptor GetMappedPropertyDescriptor(object obj, string mappedName, out object childObject, out string propertyName, bool inDesigner) | ||
|  |         { | ||
|  |             childObject = LocatePropertyObject(obj, mappedName, out propertyName, inDesigner); | ||
|  |             if (childObject == null) return null; | ||
|  |             PropertyDescriptorCollection properties = TargetFrameworkUtil.GetProperties(childObject); | ||
|  |             return properties[propertyName]; | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         /* | ||
|  |          * Walks the object model using the mapped property name to set the | ||
|  |          * value of an instance's property. | ||
|  |          */ | ||
|  |         internal static void SetMappedPropertyValue(object obj, string mappedName, object value, bool inDesigner) { | ||
|  |             string propertyName; | ||
|  |             object childObj = LocatePropertyObject(obj, mappedName, out propertyName, inDesigner); | ||
|  |             if (childObj == null) return; | ||
|  |             FastPropertyAccessor.SetProperty(childObj, propertyName, value, inDesigner); | ||
|  |         } | ||
|  | 
 | ||
|  |     } | ||
|  | } |