//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
//------------------------------------------------------------------------------
namespace System.Web.UI.WebControls {
using System;
using System.Collections;
using System.ComponentModel;
using System.Data;
using System.Drawing.Design;
using System.Globalization;
using System.Text;
using System.Web;
using System.Web.UI;
using System.Xml;
using AttributeCollection = System.ComponentModel.AttributeCollection;
///
/// Represents the data associated with a single hierarchy item - in this case an XmlNode.
///
internal sealed class XmlHierarchyData : IHierarchyData, ICustomTypeDescriptor {
private XmlNode _item;
private XmlHierarchicalEnumerable _parent;
private string _path;
///
/// Create a new instance of XmlHierarchyData.
///
internal XmlHierarchyData(XmlHierarchicalEnumerable parent, XmlNode item) {
_parent = parent;
_item = item;
}
///
/// Creates a path to a given XmlNode from the root using XPath.
///
private string CreateRecursivePath(XmlNode node) {
if (node.ParentNode == null)
return String.Empty;
return CreateRecursivePath(node.ParentNode) + FindNodePosition(node);
}
///
/// Finds a node's position relative to its parent.
///
private string FindNodePosition(XmlNode node) {
XmlNodeList nodeList = node.ParentNode.ChildNodes;
int index = 0;
for (int i = 0; i < nodeList.Count; i++) {
// Position only considers elements, not other node types
if (nodeList[i].NodeType == XmlNodeType.Element)
index++;
if (nodeList[i] == node)
return "/*[position()=" + Convert.ToString(index, CultureInfo.InvariantCulture) + "]";
}
throw new ArgumentException(SR.GetString(SR.XmlHierarchyData_CouldNotFindNode));
}
public override string ToString() {
return _item.Name;
}
bool IHierarchyData.HasChildren {
get {
return _item.HasChildNodes;
}
}
object IHierarchyData.Item {
get {
return _item;
}
}
string IHierarchyData.Path {
get {
if (_path == null) {
// If we don't yet have a path, create one and cache it
if (_parent != null) {
// If we have a parent enumerable, then our path is the parent's
// path plus our position within that parent.
if (_parent.Path == null) {
_parent.Path = CreateRecursivePath(_item.ParentNode);
}
_path = _parent.Path + FindNodePosition(_item);
}
else {
// If we are not associated with a parent enumerable, we
// have to build up our entire path from scratch.
_path = CreateRecursivePath(_item);
}
}
return _path;
}
}
string IHierarchyData.Type {
get {
return _item.Name;
}
}
IHierarchicalEnumerable IHierarchyData.GetChildren() {
return new XmlHierarchicalEnumerable(_item.ChildNodes);
}
IHierarchyData IHierarchyData.GetParent() {
XmlNode parentNode = _item.ParentNode;
if (parentNode == null)
return null;
return new XmlHierarchyData(null, parentNode);
}
AttributeCollection ICustomTypeDescriptor.GetAttributes() {
return AttributeCollection.Empty;
}
string ICustomTypeDescriptor.GetClassName() {
return GetType().Name;
}
string ICustomTypeDescriptor.GetComponentName() {
return null;
}
TypeConverter ICustomTypeDescriptor.GetConverter() {
return null;
}
EventDescriptor ICustomTypeDescriptor.GetDefaultEvent() {
return null;
}
PropertyDescriptor ICustomTypeDescriptor.GetDefaultProperty() {
return new XmlHierarchyDataPropertyDescriptor("#Name");
}
object ICustomTypeDescriptor.GetEditor(Type editorBaseType) {
return null;
}
EventDescriptorCollection ICustomTypeDescriptor.GetEvents() {
return null;
}
EventDescriptorCollection ICustomTypeDescriptor.GetEvents(Attribute[] attrs) {
return null;
}
PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties() {
return ((ICustomTypeDescriptor)this).GetProperties(null);
}
PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties(Attribute[] attrFilter) {
System.Collections.Generic.List list = new System.Collections.Generic.List();
list.Add(new XmlHierarchyDataPropertyDescriptor("#Name"));
list.Add(new XmlHierarchyDataPropertyDescriptor("#Value"));
list.Add(new XmlHierarchyDataPropertyDescriptor("#InnerText"));
XmlAttributeCollection attrs = _item.Attributes;
if (attrs != null) {
for (int i = 0; i < attrs.Count; i++) {
list.Add(new XmlHierarchyDataPropertyDescriptor(attrs[i].Name));
}
}
return new PropertyDescriptorCollection(list.ToArray());
}
object ICustomTypeDescriptor.GetPropertyOwner(PropertyDescriptor pd) {
if (pd is XmlHierarchyDataPropertyDescriptor) {
return this;
}
return null;
}
private class XmlHierarchyDataPropertyDescriptor : PropertyDescriptor {
private string _name;
public XmlHierarchyDataPropertyDescriptor(string name) : base(name, null) {
_name = name;
}
public override Type ComponentType {
get {
return typeof(XmlHierarchyData);
}
}
public override bool IsReadOnly {
get {
return true;
}
}
public override Type PropertyType {
get {
return typeof(string);
}
}
public override bool CanResetValue(object o) {
return false;
}
public override object GetValue(object o) {
XmlHierarchyData data = o as XmlHierarchyData;
if (data != null) {
switch (_name) {
case "#Name":
return data._item.Name;
case "#Value":
return data._item.Value;
case "#InnerText":
return data._item.InnerText;
default:
XmlAttributeCollection attrs = data._item.Attributes;
if (attrs != null) {
XmlAttribute attr = attrs[_name];
if (attr != null) {
return attr.Value;
}
}
break;
}
}
return String.Empty;
}
public override void ResetValue(object o) {
return;
}
public override void SetValue(object o, object value) {
}
public override bool ShouldSerializeValue(object o) {
return true;
}
}
}
}