DynamicXML.cs 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Dynamic;
  6. using System.Xml.Linq;
  7. #pragma warning disable CS8601 // Possible null reference assignment.
  8. #pragma warning disable CS8602 // Dereference of a possibly null reference.
  9. #pragma warning disable CS8603 // Possible null reference return.
  10. #pragma warning disable CS8604 // Possible null reference argument.
  11. #pragma warning disable CS8605 // Unboxing a possibly null value.
  12. #pragma warning disable CS8618 // Non-nullable property 'Text' must contain a non-null value when exiting constructor. Consider declaring the property as nullable.
  13. #pragma warning disable CS8600 // Converting null literal or possible null value to non-nullable type.
  14. namespace EasyDevCore.Common.Wrapper
  15. {
  16. /// <summary>
  17. ///
  18. /// </summary>
  19. public class DynamicXml : DynamicObject
  20. {
  21. /// <summary>
  22. /// The _root
  23. /// </summary>
  24. XElement _root;
  25. /// <summary>
  26. /// Prevents a default instance of the <see cref="DynamicXml"/> class from being created.
  27. /// </summary>
  28. /// <param name="root">The root.</param>
  29. private DynamicXml(XElement root)
  30. {
  31. _root = root;
  32. }
  33. /// <summary>
  34. /// Parses the specified XML string.
  35. /// </summary>
  36. /// <param name="xmlString">The XML string.</param>
  37. /// <returns></returns>
  38. public static DynamicXml Parse(string xmlString)
  39. {
  40. return new DynamicXml(XDocument.Parse(xmlString).Root);
  41. }
  42. /// <summary>
  43. /// Loads the specified filename.
  44. /// </summary>
  45. /// <param name="filename">The filename.</param>
  46. /// <returns></returns>
  47. public static DynamicXml Load(string filename)
  48. {
  49. return new DynamicXml(XDocument.Load(filename).Root);
  50. }
  51. /// <summary>
  52. /// Provides the implementation for operations that get member values. Classes derived from the <see cref="T:System.Dynamic.DynamicObject" /> class can override this method to specify dynamic behavior for operations such as getting a value for a property.
  53. /// </summary>
  54. /// <param name="binder">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 <see cref="T:System.Dynamic.DynamicObject" /> class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive.</param>
  55. /// <param name="result">The result of the get operation. For example, if the method is called for a property, you can assign the property value to <paramref name="result" />.</param>
  56. /// <returns>
  57. /// 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.)
  58. /// </returns>
  59. public override bool TryGetMember(GetMemberBinder binder, out object result)
  60. {
  61. result = null;
  62. var att = _root.Attribute(binder.Name);
  63. if (att != null)
  64. {
  65. result = att.Value;
  66. return true;
  67. }
  68. var nodes = _root.Elements(binder.Name);
  69. if (nodes.Count() > 1)
  70. {
  71. result = nodes.Select(n => new DynamicXml(n)).ToList();
  72. return true;
  73. }
  74. var node = _root.Element(binder.Name);
  75. if (node != null)
  76. {
  77. if (node.HasElements)
  78. {
  79. result = new DynamicXml(node);
  80. }
  81. else
  82. {
  83. result = node.Value;
  84. }
  85. return true;
  86. }
  87. return true;
  88. }
  89. }
  90. /// <summary>
  91. ///
  92. /// </summary>
  93. public class DynamicXMLNode : DynamicObject
  94. {
  95. /// <summary>
  96. /// Gets or sets the node.
  97. /// </summary>
  98. /// <value>
  99. /// The node.
  100. /// </value>
  101. public XElement Node
  102. {
  103. private set;
  104. get;
  105. }
  106. /// <summary>
  107. /// Initializes a new instance of the <see cref="DynamicXMLNode"/> class.
  108. /// </summary>
  109. /// <param name="node">The node.</param>
  110. public DynamicXMLNode(XElement node)
  111. {
  112. this.Node = node;
  113. }
  114. /// <summary>
  115. /// Initializes a new instance of the <see cref="DynamicXMLNode" /> class.
  116. /// </summary>
  117. public DynamicXMLNode()
  118. {
  119. }
  120. /// <summary>
  121. /// Initializes a new instance of the <see cref="DynamicXMLNode"/> class.
  122. /// </summary>
  123. /// <param name="name">The name.</param>
  124. public DynamicXMLNode(String name)
  125. {
  126. Node = new XElement(name);
  127. }
  128. /// <summary>
  129. /// Provides the implementation for operations that set member values. Classes derived from the <see cref="T:System.Dynamic.DynamicObject" /> class can override this method to specify dynamic behavior for operations such as setting a value for a property.
  130. /// </summary>
  131. /// <param name="binder">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 <see cref="T:System.Dynamic.DynamicObject" /> class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive.</param>
  132. /// <param name="value">The value to set to the member. For example, for sampleObject.SampleProperty = "Test", where sampleObject is an instance of the class derived from the <see cref="T:System.Dynamic.DynamicObject" /> class, the <paramref name="value" /> is "Test".</param>
  133. /// <returns>
  134. /// 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.)
  135. /// </returns>
  136. public override bool TrySetMember(SetMemberBinder binder, object value)
  137. {
  138. XElement setNode = Node.Element(binder.Name);
  139. if (setNode != null)
  140. setNode.SetValue(value);
  141. else
  142. {
  143. if (value.GetType() == typeof(DynamicXMLNode))
  144. Node.Add(new XElement(binder.Name));
  145. else
  146. Node.Add(new XElement(binder.Name, value));
  147. }
  148. return true;
  149. }
  150. /// <summary>
  151. /// Provides the implementation for operations that get member values. Classes derived from the <see cref="T:System.Dynamic.DynamicObject" /> class can override this method to specify dynamic behavior for operations such as getting a value for a property.
  152. /// </summary>
  153. /// <param name="binder">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 <see cref="T:System.Dynamic.DynamicObject" /> class, binder.Name returns "SampleProperty". The binder.IgnoreCase property specifies whether the member name is case-sensitive.</param>
  154. /// <param name="result">The result of the get operation. For example, if the method is called for a property, you can assign the property value to <paramref name="result" />.</param>
  155. /// <returns>
  156. /// 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.)
  157. /// </returns>
  158. public override bool TryGetMember(
  159. GetMemberBinder binder, out object result)
  160. {
  161. XElement getNode = Node.Element(binder.Name);
  162. if (getNode != null)
  163. {
  164. result = new DynamicXMLNode(getNode);
  165. return true;
  166. }
  167. else
  168. {
  169. result = null;
  170. return false;
  171. }
  172. }
  173. }
  174. }