using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Dynamic; using System.Xml.Linq; #pragma warning disable CS8601 // Possible null reference assignment. #pragma warning disable CS8602 // Dereference of a possibly null reference. #pragma warning disable CS8603 // Possible null reference return. #pragma warning disable CS8604 // Possible null reference argument. #pragma warning disable CS8605 // Unboxing a possibly null value. #pragma warning disable CS8618 // Non-nullable property 'Text' must contain a non-null value when exiting constructor. Consider declaring the property as nullable. #pragma warning disable CS8600 // Converting null literal or possible null value to non-nullable type. namespace EasyDevCore.Common.Wrapper { /// /// /// public class DynamicXml : DynamicObject { /// /// The _root /// XElement _root; /// /// Prevents a default instance of the class from being created. /// /// The root. private DynamicXml(XElement root) { _root = root; } /// /// Parses the specified XML string. /// /// The XML string. /// public static DynamicXml Parse(string xmlString) { return new DynamicXml(XDocument.Parse(xmlString).Root); } /// /// Loads the specified filename. /// /// The filename. /// public static DynamicXml Load(string filename) { return new DynamicXml(XDocument.Load(filename).Root); } /// /// Provides the implementation for operations that get member values. Classes derived from the class can override this method to specify dynamic behavior for operations such as getting a value for a property. /// /// Provides information about the object that called the dynamic operation. The binder.Name property provides the name of the member on which the dynamic operation is performed. For example, for the Console.WriteLine(sampleObject.SampleProperty) statement, where sampleObject is an instance of the class derived from the class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive. /// The result of the get operation. For example, if the method is called for a property, you can assign the property value to . /// /// true if the operation is successful; otherwise, false. If this method returns false, the run-time binder of the language determines the behavior. (In most cases, a run-time exception is thrown.) /// public override bool TryGetMember(GetMemberBinder binder, out object result) { result = null; var att = _root.Attribute(binder.Name); if (att != null) { result = att.Value; return true; } var nodes = _root.Elements(binder.Name); if (nodes.Count() > 1) { result = nodes.Select(n => new DynamicXml(n)).ToList(); return true; } var node = _root.Element(binder.Name); if (node != null) { if (node.HasElements) { result = new DynamicXml(node); } else { result = node.Value; } return true; } return true; } } /// /// /// public class DynamicXMLNode : DynamicObject { /// /// Gets or sets the node. /// /// /// The node. /// public XElement Node { private set; get; } /// /// Initializes a new instance of the class. /// /// The node. public DynamicXMLNode(XElement node) { this.Node = node; } /// /// Initializes a new instance of the class. /// public DynamicXMLNode() { } /// /// Initializes a new instance of the class. /// /// The name. public DynamicXMLNode(String name) { Node = new XElement(name); } /// /// Provides the implementation for operations that set member values. Classes derived from the class can override this method to specify dynamic behavior for operations such as setting a value for a property. /// /// Provides information about the object that called the dynamic operation. The binder.Name property provides the name of the member to which the value is being assigned. For example, for the statement sampleObject.SampleProperty = "Test", where sampleObject is an instance of the class derived from the class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive. /// The value to set to the member. For example, for sampleObject.SampleProperty = "Test", where sampleObject is an instance of the class derived from the class, the is "Test". /// /// true if the operation is successful; otherwise, false. If this method returns false, the run-time binder of the language determines the behavior. (In most cases, a language-specific run-time exception is thrown.) /// public override bool TrySetMember(SetMemberBinder binder, object value) { XElement setNode = Node.Element(binder.Name); if (setNode != null) setNode.SetValue(value); else { if (value.GetType() == typeof(DynamicXMLNode)) Node.Add(new XElement(binder.Name)); else Node.Add(new XElement(binder.Name, value)); } return true; } /// /// Provides the implementation for operations that get member values. Classes derived from the class can override this method to specify dynamic behavior for operations such as getting a value for a property. /// /// Provides information about the object that called the dynamic operation. The binder.Name property provides the name of the member on which the dynamic operation is performed. For example, for the Console.WriteLine(sampleObject.SampleProperty) statement, where sampleObject is an instance of the class derived from the class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive. /// The result of the get operation. For example, if the method is called for a property, you can assign the property value to . /// /// true if the operation is successful; otherwise, false. If this method returns false, the run-time binder of the language determines the behavior. (In most cases, a run-time exception is thrown.) /// public override bool TryGetMember( GetMemberBinder binder, out object result) { XElement getNode = Node.Element(binder.Name); if (getNode != null) { result = new DynamicXMLNode(getNode); return true; } else { result = null; return false; } } } }