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 sync 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; } } } } }