393 lines
15 KiB
C#
393 lines
15 KiB
C#
|
namespace System.Workflow.ComponentModel.Design
|
||
|
{
|
||
|
using System;
|
||
|
using System.Collections;
|
||
|
using System.ComponentModel;
|
||
|
using System.ComponentModel.Design;
|
||
|
using System.Diagnostics;
|
||
|
using System.Reflection;
|
||
|
using System.Globalization;
|
||
|
|
||
|
internal sealed class ReferenceService : IReferenceService, IDisposable
|
||
|
{
|
||
|
private static readonly Attribute[] Attributes = new Attribute[] { BrowsableAttribute.Yes };
|
||
|
private IServiceProvider provider; // service provider we use to get to other services
|
||
|
private ArrayList addedComponents; // list of newly added components
|
||
|
private ArrayList removedComponents; // list of newly removed components
|
||
|
private ArrayList changedComponents; // list of changed components, we will re-cylcle their references too
|
||
|
private ArrayList references; // our current list of references
|
||
|
|
||
|
internal ReferenceService(IServiceProvider provider)
|
||
|
{
|
||
|
this.provider = provider;
|
||
|
}
|
||
|
|
||
|
~ReferenceService()
|
||
|
{
|
||
|
Dispose(false);
|
||
|
}
|
||
|
|
||
|
private void CreateReferences(IComponent component)
|
||
|
{
|
||
|
CreateReferences(string.Empty, component, component);
|
||
|
}
|
||
|
|
||
|
private void CreateReferences(string trailingName, object reference, IComponent sitedComponent)
|
||
|
{
|
||
|
if (object.ReferenceEquals(reference, null))
|
||
|
return;
|
||
|
|
||
|
this.references.Add(new ReferenceHolder(trailingName, reference, sitedComponent));
|
||
|
foreach (PropertyDescriptor property in TypeDescriptor.GetProperties(reference, Attributes))
|
||
|
{
|
||
|
object value = null;
|
||
|
try
|
||
|
{
|
||
|
value = property.GetValue(reference);
|
||
|
}
|
||
|
catch
|
||
|
{
|
||
|
// Work around!!! if an property getter throws exception then we ignore it.
|
||
|
}
|
||
|
if (value != null)
|
||
|
{
|
||
|
BrowsableAttribute[] browsableAttrs = (BrowsableAttribute[])(value.GetType().GetCustomAttributes(typeof(BrowsableAttribute), true));
|
||
|
if (browsableAttrs.Length > 0 && browsableAttrs[0].Browsable)
|
||
|
{
|
||
|
CreateReferences(string.Format(CultureInfo.InvariantCulture, "{0}.{1}", new object[] { trailingName, property.Name }), property.GetValue(reference), sitedComponent);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void EnsureReferences()
|
||
|
{
|
||
|
// If the references are null, create them for the first time and connect
|
||
|
// up our events to listen to changes to the container. Otherwise, check to
|
||
|
// see if the added or removed lists contain anything for us to [....] up.
|
||
|
//
|
||
|
if (this.references == null)
|
||
|
{
|
||
|
if (this.provider == null)
|
||
|
{
|
||
|
throw new ObjectDisposedException("IReferenceService");
|
||
|
}
|
||
|
|
||
|
IComponentChangeService cs = this.provider.GetService(typeof(IComponentChangeService)) as IComponentChangeService;
|
||
|
Debug.Assert(cs != null, "Reference service relies on IComponentChangeService");
|
||
|
if (cs != null)
|
||
|
{
|
||
|
cs.ComponentAdded += new ComponentEventHandler(this.OnComponentAdded);
|
||
|
cs.ComponentRemoved += new ComponentEventHandler(this.OnComponentRemoved);
|
||
|
cs.ComponentRename += new ComponentRenameEventHandler(this.OnComponentRename);
|
||
|
cs.ComponentChanged += new ComponentChangedEventHandler(this.OnComponentChanged);
|
||
|
}
|
||
|
|
||
|
TypeDescriptor.Refreshed += new RefreshEventHandler(OnComponentRefreshed);
|
||
|
|
||
|
IContainer container = this.provider.GetService(typeof(IContainer)) as IContainer;
|
||
|
if (container == null)
|
||
|
{
|
||
|
Debug.Fail("Reference service cannot operate without IContainer");
|
||
|
throw new InvalidOperationException();
|
||
|
}
|
||
|
|
||
|
this.references = new ArrayList(container.Components.Count);
|
||
|
|
||
|
foreach (IComponent component in container.Components)
|
||
|
{
|
||
|
CreateReferences(component);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (this.addedComponents != null && this.addedComponents.Count > 0)
|
||
|
{
|
||
|
// There is a possibility that this component already exists.
|
||
|
// If it does, just remove it first and then re-add it.
|
||
|
//
|
||
|
ArrayList clonedAddedComponents = new ArrayList(this.addedComponents);
|
||
|
foreach (IComponent ic in clonedAddedComponents)
|
||
|
{
|
||
|
RemoveReferences(ic);
|
||
|
CreateReferences(ic);
|
||
|
}
|
||
|
this.addedComponents.Clear();
|
||
|
}
|
||
|
|
||
|
if (this.removedComponents != null && this.removedComponents.Count > 0)
|
||
|
{
|
||
|
ArrayList clonedRemovedComponents = new ArrayList(this.removedComponents);
|
||
|
foreach (IComponent ic in clonedRemovedComponents)
|
||
|
RemoveReferences(ic);
|
||
|
|
||
|
this.removedComponents.Clear();
|
||
|
}
|
||
|
if (this.changedComponents != null && this.changedComponents.Count > 0)
|
||
|
{
|
||
|
ArrayList clonedChangedComponents = new ArrayList(this.changedComponents);
|
||
|
foreach (IComponent ic in clonedChangedComponents)
|
||
|
{
|
||
|
RemoveReferences(ic);
|
||
|
CreateReferences(ic);
|
||
|
}
|
||
|
this.changedComponents.Clear();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void OnComponentChanged(object sender, ComponentChangedEventArgs cevent)
|
||
|
{
|
||
|
IComponent comp = ((IReferenceService)this).GetComponent(cevent.Component);
|
||
|
|
||
|
if (comp != null)
|
||
|
{
|
||
|
if ((this.addedComponents == null || !this.addedComponents.Contains(comp)) &&
|
||
|
(this.removedComponents == null || !this.removedComponents.Contains(comp)))
|
||
|
{
|
||
|
if (this.changedComponents == null)
|
||
|
{
|
||
|
this.changedComponents = new ArrayList();
|
||
|
this.changedComponents.Add(comp);
|
||
|
}
|
||
|
else if (!this.changedComponents.Contains(comp))
|
||
|
{
|
||
|
this.changedComponents.Add(comp);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private void OnComponentAdded(object sender, ComponentEventArgs cevent)
|
||
|
{
|
||
|
if (this.addedComponents == null)
|
||
|
this.addedComponents = new ArrayList();
|
||
|
this.addedComponents.Add(cevent.Component);
|
||
|
if (this.removedComponents != null)
|
||
|
this.removedComponents.Remove(cevent.Component);
|
||
|
if (this.changedComponents != null)
|
||
|
this.changedComponents.Remove(cevent.Component);
|
||
|
}
|
||
|
|
||
|
private void OnComponentRemoved(object sender, ComponentEventArgs cevent)
|
||
|
{
|
||
|
if (this.removedComponents == null)
|
||
|
this.removedComponents = new ArrayList();
|
||
|
|
||
|
this.removedComponents.Add(cevent.Component);
|
||
|
|
||
|
if (this.addedComponents != null)
|
||
|
this.addedComponents.Remove(cevent.Component);
|
||
|
if (this.changedComponents != null)
|
||
|
this.changedComponents.Remove(cevent.Component);
|
||
|
}
|
||
|
|
||
|
private void OnComponentRename(object sender, ComponentRenameEventArgs cevent)
|
||
|
{
|
||
|
foreach (ReferenceHolder reference in this.references)
|
||
|
{
|
||
|
if (object.ReferenceEquals(reference.SitedComponent, cevent.Component))
|
||
|
{
|
||
|
reference.ResetName();
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
private void OnComponentRefreshed(RefreshEventArgs e)
|
||
|
{
|
||
|
if (e.ComponentChanged != null)
|
||
|
OnComponentChanged(this, new ComponentChangedEventArgs(e.ComponentChanged, null, null, null));
|
||
|
}
|
||
|
|
||
|
private void RemoveReferences(IComponent component)
|
||
|
{
|
||
|
if (this.references != null)
|
||
|
{
|
||
|
int size = this.references.Count;
|
||
|
for (int i = size - 1; i >= 0; i--)
|
||
|
{
|
||
|
if (object.ReferenceEquals(((ReferenceHolder)this.references[i]).SitedComponent, component))
|
||
|
this.references.RemoveAt(i);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Dispose()
|
||
|
{
|
||
|
Dispose(true);
|
||
|
GC.SuppressFinalize(this);
|
||
|
}
|
||
|
|
||
|
private void Dispose(bool disposing)
|
||
|
{
|
||
|
if (this.references != null && this.provider != null)
|
||
|
{
|
||
|
IComponentChangeService cs = this.provider.GetService(typeof(IComponentChangeService)) as IComponentChangeService;
|
||
|
if (cs != null)
|
||
|
{
|
||
|
cs.ComponentAdded -= new ComponentEventHandler(this.OnComponentAdded);
|
||
|
cs.ComponentRemoved -= new ComponentEventHandler(this.OnComponentRemoved);
|
||
|
cs.ComponentRename -= new ComponentRenameEventHandler(this.OnComponentRename);
|
||
|
cs.ComponentChanged -= new ComponentChangedEventHandler(this.OnComponentChanged);
|
||
|
}
|
||
|
|
||
|
TypeDescriptor.Refreshed -= new RefreshEventHandler(OnComponentRefreshed);
|
||
|
|
||
|
this.references = null;
|
||
|
this.provider = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
IComponent IReferenceService.GetComponent(object reference)
|
||
|
{
|
||
|
if (object.ReferenceEquals(reference, null))
|
||
|
throw new ArgumentNullException("reference");
|
||
|
|
||
|
EnsureReferences();
|
||
|
|
||
|
foreach (ReferenceHolder holder in this.references)
|
||
|
{
|
||
|
if (object.ReferenceEquals(holder.Reference, reference))
|
||
|
return holder.SitedComponent;
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
string IReferenceService.GetName(object reference)
|
||
|
{
|
||
|
if (object.ReferenceEquals(reference, null))
|
||
|
throw new ArgumentNullException("reference");
|
||
|
|
||
|
EnsureReferences();
|
||
|
|
||
|
foreach (ReferenceHolder holder in this.references)
|
||
|
{
|
||
|
if (object.ReferenceEquals(holder.Reference, reference))
|
||
|
return holder.Name;
|
||
|
}
|
||
|
return null;
|
||
|
}
|
||
|
object IReferenceService.GetReference(string name)
|
||
|
{
|
||
|
if (name == null)
|
||
|
throw new ArgumentNullException("name");
|
||
|
|
||
|
EnsureReferences();
|
||
|
|
||
|
foreach (ReferenceHolder holder in this.references)
|
||
|
{
|
||
|
if (string.Equals(holder.Name, name, StringComparison.OrdinalIgnoreCase))
|
||
|
return holder.Reference;
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
object[] IReferenceService.GetReferences()
|
||
|
{
|
||
|
EnsureReferences();
|
||
|
|
||
|
object[] references = new object[this.references.Count];
|
||
|
for (int i = 0; i < references.Length; i++)
|
||
|
references[i] = ((ReferenceHolder)this.references[i]).Reference;
|
||
|
|
||
|
return references;
|
||
|
}
|
||
|
|
||
|
object[] IReferenceService.GetReferences(Type baseType)
|
||
|
{
|
||
|
if (baseType == null)
|
||
|
throw new ArgumentNullException("baseType");
|
||
|
|
||
|
EnsureReferences();
|
||
|
|
||
|
ArrayList results = new ArrayList(this.references.Count);
|
||
|
foreach (ReferenceHolder holder in this.references)
|
||
|
{
|
||
|
object reference = holder.Reference;
|
||
|
if (baseType.IsAssignableFrom(reference.GetType()))
|
||
|
results.Add(reference);
|
||
|
}
|
||
|
|
||
|
object[] references = new object[results.Count];
|
||
|
results.CopyTo(references, 0);
|
||
|
return references;
|
||
|
}
|
||
|
|
||
|
private sealed class ReferenceHolder
|
||
|
{
|
||
|
private string trailingName;
|
||
|
private object reference;
|
||
|
private IComponent sitedComponent;
|
||
|
private string fullName;
|
||
|
|
||
|
internal ReferenceHolder(string trailingName, object reference, IComponent sitedComponent)
|
||
|
{
|
||
|
this.trailingName = trailingName;
|
||
|
this.reference = reference;
|
||
|
this.sitedComponent = sitedComponent;
|
||
|
|
||
|
Debug.Assert(trailingName != null, "Expected a trailing name");
|
||
|
Debug.Assert(reference != null, "Expected a reference");
|
||
|
|
||
|
#if DEBUG
|
||
|
|
||
|
Debug.Assert(sitedComponent != null, "Expected a sited component");
|
||
|
if (sitedComponent != null)
|
||
|
Debug.Assert(sitedComponent.Site != null, "Sited component is not really sited: " + sitedComponent.ToString());
|
||
|
if (sitedComponent != null && sitedComponent.Site != null)
|
||
|
Debug.Assert(sitedComponent.Site.Name != null, "Sited component has no name: " + sitedComponent.ToString());
|
||
|
|
||
|
#endif // DEBUG
|
||
|
}
|
||
|
|
||
|
internal void ResetName()
|
||
|
{
|
||
|
this.fullName = null;
|
||
|
}
|
||
|
|
||
|
internal string Name
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
if (this.fullName == null)
|
||
|
{
|
||
|
if (this.sitedComponent != null && this.sitedComponent.Site != null && this.sitedComponent.Site.Name != null)
|
||
|
{
|
||
|
this.fullName = string.Format(CultureInfo.InvariantCulture, "{0}{1}", new object[] { this.sitedComponent.Site.Name, this.trailingName });
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
#if DEBUG
|
||
|
Debug.Assert(this.sitedComponent != null, "Expected a sited component");
|
||
|
if (this.sitedComponent != null)
|
||
|
Debug.Assert(this.sitedComponent.Site != null, "Sited component is not really sited: " + this.sitedComponent.ToString());
|
||
|
if (this.sitedComponent != null && this.sitedComponent.Site != null)
|
||
|
Debug.Assert(this.sitedComponent.Site.Name != null, "Sited component has no name: " + this.sitedComponent.ToString());
|
||
|
#endif // DEBUG
|
||
|
this.fullName = string.Empty;
|
||
|
}
|
||
|
}
|
||
|
return this.fullName;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal object Reference
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
return this.reference;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal IComponent SitedComponent
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
return this.sitedComponent;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|