You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			575 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			575 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------------------------
 | |
| // <copyright file="ReflectEventDescriptor.cs" company="Microsoft">
 | |
| //     Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>                                                                
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| 
 | |
| /*
 | |
|  */
 | |
| namespace System.ComponentModel {
 | |
|     using Microsoft.Win32;
 | |
|     using System;
 | |
|     using System.Collections;
 | |
|     using System.ComponentModel.Design;
 | |
|     using System.Diagnostics;
 | |
|     using System.Reflection;
 | |
|     using System.Security.Permissions;
 | |
| 
 | |
|     /// <internalonly/>
 | |
|     /// <devdoc>
 | |
|     ///    <para>
 | |
|     ///       ReflectEventDescriptor defines an event. Events are the main way that a user can get
 | |
|     ///       run-time notifications from a component.
 | |
|     ///       The ReflectEventDescriptor class takes a component class that the event lives on,
 | |
|     ///       the event name, the type of the event handling delegate, and various
 | |
|     ///       attributes for the event.
 | |
|     ///       Every event has a structure through which it passes it's information. The base
 | |
|     ///       structure, Event, is empty, and there is a default instance, Event.EMPTY, which
 | |
|     ///       is usually passed. When addOnXXX is invoked, it needs a pointer to a method
 | |
|     ///       that takes a source object (the object that fired the event) and a structure
 | |
|     ///       particular to that event. It also needs a pointer to the instance of the
 | |
|     ///       object the method should be invoked on. These two things are what composes a
 | |
|     ///       delegate. An event handler is
 | |
|     ///       a delegate, and the compiler recognizes a special delegate syntax that makes
 | |
|     ///       using delegates easy.
 | |
|     ///       For example, to listen to the click event on a button in class Foo, the
 | |
|     ///       following code will suffice:
 | |
|     ///    </para>
 | |
|     ///    <code>
 | |
|     /// class Foo {
 | |
|     ///     Button button1 = new Button();
 | |
|     ///     void button1_click(Object sender, Event e) {
 | |
|     ///     // do something on button1 click.
 | |
|     ///     }
 | |
|     ///     public Foo() {
 | |
|     ///     button1.addOnClick(this.button1_click);
 | |
|     ///     }
 | |
|     ///     }
 | |
|     ///    </code>
 | |
|     ///    For an event named XXX, a YYYEvent structure, and a YYYEventHandler delegate,
 | |
|     ///    a component writer is required to implement two methods of the following
 | |
|     ///    form:
 | |
|     ///    <code>
 | |
|     /// public void addOnXXX(YYYEventHandler handler);
 | |
|     ///     public void removeOnXXX(YYYEventHandler handler);
 | |
|     ///    </code>
 | |
|     ///    YYYEventHandler should be an event handler declared as
 | |
|     ///    <code>
 | |
|     /// public multicast delegate void YYYEventHandler(Object sender, YYYEvent e);
 | |
|     ///    </code>
 | |
|     ///    Note that this event was declared as a multicast delegate. This allows multiple
 | |
|     ///    listeners on an event. This is not a requirement.
 | |
|     ///    Various attributes can be passed to the ReflectEventDescriptor, as are described in
 | |
|     ///    Attribute.
 | |
|     ///    ReflectEventDescriptors can be obtained by a user programmatically through the
 | |
|     ///    ComponentManager.
 | |
|     /// </devdoc>
 | |
|     [HostProtection(SharedState = true)]
 | |
|     internal sealed class ReflectEventDescriptor : EventDescriptor {
 | |
| #if !MONO
 | |
|         private static readonly Type[] argsNone = new Type[0];
 | |
|         private static readonly object  noDefault = new object();
 | |
| #endif
 | |
| 
 | |
|         private Type type;           // the delegate type for the event
 | |
|         private readonly Type componentClass; // the class of the component this info is for
 | |
| 
 | |
|         private MethodInfo addMethod;     // the method to use when adding an event
 | |
|         private MethodInfo removeMethod;  // the method to use when removing an event
 | |
|         private EventInfo realEvent;      // actual event info... may be null
 | |
|         private bool    filledMethods = false;   // did we already call FillMethods() once?
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is the main constructor for an ReflectEventDescriptor.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type,
 | |
|                                       Attribute[] attributes)
 | |
|         : base(name, attributes) {
 | |
|             if (componentClass == null) {
 | |
|                 throw new ArgumentException(SR.GetString(SR.InvalidNullArgument, "componentClass"));
 | |
|             }
 | |
|             if (type == null || !(typeof(Delegate)).IsAssignableFrom(type)) {
 | |
|                 throw new ArgumentException(SR.GetString(SR.ErrorInvalidEventType, name));
 | |
|             }
 | |
|             Debug.Assert(type.IsSubclassOf(typeof(Delegate)), "Not a valid ReflectEvent: " + componentClass.FullName + "." + name + " " + type.FullName);
 | |
|             this.componentClass = componentClass;
 | |
|             this.type = type;
 | |
|         }
 | |
| 
 | |
|         public ReflectEventDescriptor(Type componentClass, EventInfo eventInfo)
 | |
|         : base(eventInfo.Name, new Attribute[0]) {
 | |
| 
 | |
|             if (componentClass == null) {
 | |
|                 throw new ArgumentException(SR.GetString(SR.InvalidNullArgument, "componentClass"));
 | |
|             }
 | |
|             this.componentClass = componentClass;
 | |
|             this.realEvent = eventInfo;
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This constructor takes an existing ReflectEventDescriptor and modifies it by merging in the
 | |
|         ///     passed-in attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentType, EventDescriptor oldReflectEventDescriptor, Attribute[] attributes)
 | |
|         : base(oldReflectEventDescriptor, attributes) {
 | |
|             this.componentClass = componentType;
 | |
|             this.type = oldReflectEventDescriptor.EventType;
 | |
|             
 | |
|             ReflectEventDescriptor desc = oldReflectEventDescriptor as ReflectEventDescriptor;
 | |
|             if (desc != null) {
 | |
|                 this.addMethod = desc.addMethod;
 | |
|                 this.removeMethod = desc.removeMethod;
 | |
|                 this.filledMethods = true;
 | |
|             }
 | |
|             #if DEBUG
 | |
|             else if (oldReflectEventDescriptor is DebugReflectEventDescriptor)
 | |
|             {
 | |
|                 this.addMethod = ((DebugReflectEventDescriptor)oldReflectEventDescriptor).addMethod;
 | |
|                 this.removeMethod = ((DebugReflectEventDescriptor)oldReflectEventDescriptor).removeMethod;
 | |
|                 this.filledMethods = true;
 | |
|             }
 | |
|             #endif
 | |
|         }
 | |
| 
 | |
| 
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     Retrieves the type of the component this EventDescriptor is bound to.
 | |
|         /// </devdoc>
 | |
|         public override Type ComponentType {
 | |
|             get {
 | |
|                 return componentClass;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     Retrieves the type of the delegate for this event.
 | |
|         /// </devdoc>
 | |
|         public override Type EventType {
 | |
|             get {
 | |
|                 FillMethods();
 | |
|                 return type;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     Indicates whether the delegate type for this event is a multicast
 | |
|         ///     delegate.
 | |
|         /// </devdoc>
 | |
|         public override bool IsMulticast {
 | |
|             get {
 | |
|                 return(typeof(MulticastDelegate)).IsAssignableFrom(EventType);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This adds the delegate value as a listener to when this event is fired
 | |
|         ///     by the component, invoking the addOnXXX method.
 | |
|         /// </devdoc>
 | |
|         public override void AddEventHandler(object component, Delegate value) {
 | |
|             FillMethods();
 | |
| 
 | |
|             if (component != null) {
 | |
|                 ISite site = GetSite(component);
 | |
|                 IComponentChangeService changeService = null;
 | |
| 
 | |
|                 // Announce that we are about to change this component
 | |
|                 //
 | |
|                 if (site != null) {
 | |
|                     changeService = (IComponentChangeService)site.GetService(typeof(IComponentChangeService));
 | |
|                     Debug.Assert(!CompModSwitches.CommonDesignerServices.Enabled || changeService != null, "IComponentChangeService not found");
 | |
|                 }
 | |
| 
 | |
|                 if (changeService != null) {
 | |
|                     try {
 | |
|                         changeService.OnComponentChanging(component, this);
 | |
|                     }
 | |
|                     catch (CheckoutException coEx) {
 | |
|                         if (coEx == CheckoutException.Canceled) {
 | |
|                             return;
 | |
|                         }
 | |
|                         throw coEx;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 bool shadowed = false;
 | |
| 
 | |
|                 if (site != null && site.DesignMode) {
 | |
|                     // Events are final, so just check the class
 | |
|                     if (EventType != value.GetType()) {
 | |
|                         throw new ArgumentException(SR.GetString(SR.ErrorInvalidEventHandler, Name));
 | |
|                     }
 | |
|                     IDictionaryService dict = (IDictionaryService)site.GetService(typeof(IDictionaryService));
 | |
|                     Debug.Assert(!CompModSwitches.CommonDesignerServices.Enabled || dict != null, "IDictionaryService not found");
 | |
|                     if (dict != null) {
 | |
|                         Delegate eventdesc = (Delegate)dict.GetValue(this);
 | |
|                         eventdesc = Delegate.Combine(eventdesc, value);
 | |
|                         dict.SetValue(this, eventdesc);
 | |
|                         shadowed = true;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (!shadowed) {
 | |
|                     SecurityUtils.MethodInfoInvoke(addMethod, component, new object[] { value});
 | |
|                 }
 | |
| 
 | |
|                 // Now notify the change service that the change was successful.
 | |
|                 //
 | |
|                 if (changeService != null) {
 | |
|                     changeService.OnComponentChanged(component, this, null, value);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // <doc>
 | |
|         // <desc>
 | |
|         //     Adds in custom attributes found on either the AddOn or RemoveOn method...
 | |
|         // </desc>
 | |
|         // </doc>
 | |
|         //
 | |
|         protected override void FillAttributes(IList attributes) {
 | |
|         
 | |
|             //
 | |
|             // The order that we fill in attributes is critical.  The list of attributes will be
 | |
|             // filtered so that matching attributes at the end of the list replace earlier matches
 | |
|             // (last one in wins).  Therefore, the two categories of attributes we add must be
 | |
|             // added as follows:
 | |
|             //
 | |
|             // 1.  Attributes of the event, from base class to most derived.  This way
 | |
|             //     derived class attributes replace base class attributes.
 | |
|             //
 | |
|             // 2.  Attributes from our base MemberDescriptor.  While this seems opposite of what
 | |
|             //     we want, MemberDescriptor only has attributes if someone passed in a new
 | |
|             //     set in the constructor.  Therefore, these attributes always
 | |
|             //     supercede existing values.
 | |
|             //
 | |
|         
 | |
|             FillMethods();
 | |
|             Debug.Assert(componentClass != null, "Must have a component class for FilterAttributes");
 | |
|             if (realEvent != null) {
 | |
|                 FillEventInfoAttribute(realEvent, attributes);
 | |
|             }
 | |
|             else {
 | |
|                 Debug.Assert(removeMethod != null, "Null remove method for " + Name);
 | |
|                 FillSingleMethodAttribute(removeMethod, attributes);
 | |
| 
 | |
|                 Debug.Assert(addMethod != null, "Null remove method for " + Name);
 | |
|                 FillSingleMethodAttribute(addMethod, attributes);
 | |
|             }
 | |
|         
 | |
|             // Include the base attributes.  These override all attributes on the actual
 | |
|             // property, so we want to add them last.
 | |
|             //
 | |
|             base.FillAttributes(attributes);
 | |
|         }
 | |
|         
 | |
|         private void FillEventInfoAttribute(EventInfo realEventInfo, IList attributes) {
 | |
|             string eventName = realEventInfo.Name;
 | |
|             BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.DeclaredOnly;
 | |
|             Type currentReflectType = realEventInfo.ReflectedType;
 | |
|             Debug.Assert(currentReflectType != null, "currentReflectType cannot be null");
 | |
|             int depth = 0;
 | |
|             
 | |
|             // First, calculate the depth of the object hierarchy.  We do this so we can do a single
 | |
|             // object create for an array of attributes.
 | |
|             //
 | |
|             while(currentReflectType != typeof(object)) {
 | |
|                 depth++;
 | |
|                 currentReflectType = currentReflectType.BaseType;
 | |
|             }
 | |
| 
 | |
|             if (depth > 0) {
 | |
|                 // Now build up an array in reverse order
 | |
|                 //
 | |
|                 currentReflectType = realEventInfo.ReflectedType;
 | |
|                 Attribute[][] attributeStack = new Attribute[depth][];
 | |
|                 
 | |
|                 while(currentReflectType != typeof(object)) {
 | |
| 
 | |
|                     // Fill in our member info so we can get at the custom attributes.
 | |
|                     //
 | |
|                     MemberInfo memberInfo = currentReflectType.GetEvent(eventName, bindingFlags);
 | |
|                     
 | |
|                     // Get custom attributes for the member info.
 | |
|                     //
 | |
|                     if (memberInfo != null) {
 | |
|                         attributeStack[--depth] = ReflectTypeDescriptionProvider.ReflectGetAttributes(memberInfo);
 | |
|                     }
 | |
|                     
 | |
|                     // Ready for the next loop iteration.
 | |
|                     //
 | |
|                     currentReflectType = currentReflectType.BaseType;
 | |
|                 }
 | |
|                 
 | |
|                 // Now trawl the attribute stack so that we add attributes
 | |
|                 // from base class to most derived.
 | |
|                 //
 | |
|                 foreach(Attribute[] attributeArray in attributeStack) {
 | |
|                     if (attributeArray != null)
 | |
|                     {
 | |
|                         foreach(Attribute attr in attributeArray) {
 | |
|                             attributes.Add(attr);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|             
 | |
|         /// <devdoc>
 | |
|         ///     This fills the get and set method fields of the event info.  It is shared
 | |
|         ///     by the various constructors.
 | |
|         /// </devdoc>
 | |
|         private void FillMethods() {
 | |
|             if (filledMethods) return;
 | |
| 
 | |
|             if (realEvent != null) {
 | |
|                 addMethod = realEvent.GetAddMethod();
 | |
|                 removeMethod = realEvent.GetRemoveMethod();
 | |
| 
 | |
|                 EventInfo defined = null;
 | |
| 
 | |
|                 if (addMethod == null || removeMethod == null) {
 | |
|                     Type start = componentClass.BaseType;
 | |
|                     while (start != null && start != typeof(object)) {
 | |
|                         BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic;
 | |
|                         EventInfo test = start.GetEvent(realEvent.Name, bindingFlags);
 | |
|                         if (test.GetAddMethod() != null) {
 | |
|                             defined = test;
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (defined != null) {
 | |
|                     addMethod = defined.GetAddMethod();
 | |
|                     removeMethod = defined.GetRemoveMethod();
 | |
|                     type = defined.EventHandlerType;
 | |
|                 }
 | |
|                 else {
 | |
|                     type = realEvent.EventHandlerType;
 | |
|                 }
 | |
|             }
 | |
|             else {
 | |
| 
 | |
|                 // first, try to get the eventInfo...
 | |
|                 //
 | |
|                 realEvent = this.componentClass.GetEvent(Name);
 | |
|                 if (realEvent != null) {
 | |
|                     // if we got one, just recurse and return.
 | |
|                     //
 | |
|                     FillMethods();
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 Type[] argsType = new Type[] {type};
 | |
|                 addMethod = FindMethod(componentClass, "AddOn" + Name, argsType, typeof(void));
 | |
|                 removeMethod = FindMethod(componentClass, "RemoveOn" + Name, argsType, typeof(void));
 | |
|                 if (addMethod == null || removeMethod == null) {
 | |
|                     Debug.Fail("Missing event accessors for " + componentClass.FullName + "." + Name);
 | |
|                     throw new ArgumentException(SR.GetString(SR.ErrorMissingEventAccessors, Name));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             filledMethods = true;
 | |
|         }
 | |
| 
 | |
|         private void FillSingleMethodAttribute(MethodInfo realMethodInfo, IList attributes) {
 | |
| 
 | |
|             string methodName = realMethodInfo.Name;
 | |
|             BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.DeclaredOnly;
 | |
|             Type currentReflectType = realMethodInfo.ReflectedType;
 | |
|             Debug.Assert(currentReflectType != null, "currentReflectType cannot be null");
 | |
| 
 | |
|             // First, calculate the depth of the object hierarchy.  We do this so we can do a single
 | |
|             // object create for an array of attributes.
 | |
|             //
 | |
|             int depth = 0;
 | |
|             while(currentReflectType != null && currentReflectType != typeof(object)) {
 | |
|                 depth++;
 | |
|                 currentReflectType = currentReflectType.BaseType;
 | |
|             }
 | |
|             
 | |
|             if (depth > 0) {
 | |
|                 // Now build up an array in reverse order
 | |
|                 //
 | |
|                 currentReflectType = realMethodInfo.ReflectedType;
 | |
|                 Attribute[][] attributeStack = new Attribute[depth][];
 | |
|                 
 | |
|                 while(currentReflectType != null && currentReflectType != typeof(object)) {
 | |
|                     // Fill in our member info so we can get at the custom attributes.
 | |
|                     //
 | |
|                     MemberInfo memberInfo = currentReflectType.GetMethod(methodName, bindingFlags);
 | |
|                     
 | |
|                     // Get custom attributes for the member info.
 | |
|                     //
 | |
|                     if (memberInfo != null) {
 | |
|                         attributeStack[--depth] = ReflectTypeDescriptionProvider.ReflectGetAttributes(memberInfo);
 | |
|                     }
 | |
|                     
 | |
|                     // Ready for the next loop iteration.
 | |
|                     //
 | |
|                     currentReflectType = currentReflectType.BaseType;
 | |
|                 }
 | |
|                 
 | |
|                 // Now trawl the attribute stack so that we add attributes
 | |
|                 // from base class to most derived.
 | |
|                 //
 | |
|                 foreach(Attribute[] attributeArray in attributeStack) {
 | |
|                     if (attributeArray != null)
 | |
|                     {
 | |
|                         foreach(Attribute attr in attributeArray) {
 | |
|                             attributes.Add(attr);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This will remove the delegate value from the event chain so that
 | |
|         ///     it no longer gets events from this component.
 | |
|         /// </devdoc>
 | |
|         public override void RemoveEventHandler(object component, Delegate value) {
 | |
|             FillMethods();
 | |
| 
 | |
|             if (component != null) {
 | |
|                 ISite site = GetSite(component);
 | |
|                 IComponentChangeService changeService = null;
 | |
| 
 | |
|                 // Announce that we are about to change this component
 | |
|                 //
 | |
|                 if (site != null) {
 | |
|                     changeService = (IComponentChangeService)site.GetService(typeof(IComponentChangeService));
 | |
|                     Debug.Assert(!CompModSwitches.CommonDesignerServices.Enabled || changeService != null, "IComponentChangeService not found");
 | |
|                 }
 | |
| 
 | |
|                 if (changeService != null) {
 | |
|                     try {
 | |
|                         changeService.OnComponentChanging(component, this);
 | |
|                     }
 | |
|                     catch (CheckoutException coEx) {
 | |
|                         if (coEx == CheckoutException.Canceled) {
 | |
|                             return;
 | |
|                         }
 | |
|                         throw coEx;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 bool shadowed = false;
 | |
| 
 | |
|                 if (site != null && site.DesignMode) {
 | |
|                     IDictionaryService dict = (IDictionaryService)site.GetService(typeof(IDictionaryService));
 | |
|                     Debug.Assert(!CompModSwitches.CommonDesignerServices.Enabled || dict != null, "IDictionaryService not found");
 | |
|                     if (dict != null) {
 | |
|                         Delegate del = (Delegate)dict.GetValue(this);
 | |
|                         del = Delegate.Remove(del, value);
 | |
|                         dict.SetValue(this, del);
 | |
|                         shadowed = true;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (!shadowed) {
 | |
|                     SecurityUtils.MethodInfoInvoke(removeMethod, component, new object[] { value });
 | |
|                 }
 | |
| 
 | |
|                 // Now notify the change service that the change was successful.
 | |
|                 //
 | |
|                 if (changeService != null) {
 | |
|                     changeService.OnComponentChanged(component, this, null, value);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /* The following code has been removed to fix FXCOP violations.
 | |
|            its left here incase it needs to be resurected. This code is from
 | |
|            ReflectEventDescriptor class
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with one attribute.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type, MethodInfo addMethod, MethodInfo removeMethod) : this(componentClass, name, type, (Attribute[]) null) {
 | |
|             this.addMethod = addMethod;
 | |
|             this.removeMethod = removeMethod;
 | |
|             this.filledMethods = true;
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with one attribute.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type) : this(componentClass, name, type, (Attribute[]) null) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with two attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type,
 | |
|                                       Attribute a1) : this(componentClass, name, type, new Attribute[] {a1}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with two attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type,
 | |
|                                       Attribute a1, Attribute a2) : this(componentClass, name, type, new Attribute[] {a1, a2}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with three attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type,
 | |
|                                       Attribute a1, Attribute a2, Attribute a3) : this(componentClass, name, type, new Attribute[] {a1, a2, a3}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut main constructor for an ReflectEventDescriptor with four attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(Type componentClass, string name, Type type,
 | |
|                                       Attribute a1, Attribute a2,
 | |
|                                       Attribute a3, Attribute a4) : this(componentClass, name, type, new Attribute[] {a1, a2, a3, a4}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This constructor takes an existing ReflectEventDescriptor and modifies it by merging in the
 | |
|         ///     passed-in attributes.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(EventDescriptor oldReflectEventDescriptor, Attribute[] attributes)
 | |
|         : this(oldReflectEventDescriptor.ComponentType, oldReflectEventDescriptor, attributes) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut constructor that takes an existing ReflectEventDescriptor and one attribute to
 | |
|         ///     merge in.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(EventDescriptor oldReflectEventDescriptor, Attribute a1) : this(oldReflectEventDescriptor, new Attribute[] { a1}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut constructor that takes an existing ReflectEventDescriptor and two attributes to
 | |
|         ///     merge in.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(EventDescriptor oldReflectEventDescriptor, Attribute a1,
 | |
|                                       Attribute a2) : this(oldReflectEventDescriptor, new Attribute[] { a1,a2}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut constructor that takes an existing ReflectEventDescriptor and three attributes to
 | |
|         ///     merge in.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(EventDescriptor oldReflectEventDescriptor, Attribute a1,
 | |
|                                       Attribute a2, Attribute a3) : this(oldReflectEventDescriptor, new Attribute[] { a1,a2,a3}) {
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     This is a shortcut constructor that takes an existing ReflectEventDescriptor and four attributes to
 | |
|         ///     merge in.
 | |
|         /// </devdoc>
 | |
|         public ReflectEventDescriptor(EventDescriptor oldReflectEventDescriptor, Attribute a1,
 | |
|                                       Attribute a2, Attribute a3, Attribute a4) : this(oldReflectEventDescriptor, new Attribute[] { a1,a2,a3,a4}) {
 | |
|         }
 | |
|         */
 | |
| 
 | |
|     }
 | |
| }
 |