XmlDataDocument.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658
  1. //
  2. // mcs/class/System.Data/System.Xml/XmlDataDocument.cs
  3. //
  4. // Purpose: Provides a W3C XML DOM Document to interact with
  5. // relational data in a DataSet
  6. //
  7. // class: XmlDataDocument
  8. // assembly: System.Data.dll
  9. // namespace: System.Xml
  10. //
  11. // Author:
  12. // Daniel Morgan <[email protected]>
  13. // Ville Palo <[email protected]>
  14. //
  15. // (c)copyright 2002 Daniel Morgan
  16. // (c)copyright 2003 Ville Palo
  17. //
  18. // XmlDataDocument is included within the Mono Class Library.
  19. //
  20. using System;
  21. using System.Data;
  22. using System.IO;
  23. using System.Text;
  24. using System.Xml.XPath;
  25. using System.Collections;
  26. using System.Globalization;
  27. using System.ComponentModel;
  28. namespace System.Xml {
  29. public class XmlDataDocument : XmlDocument {
  30. #region Fields
  31. private DataSet dataSet;
  32. private bool isReadOnly = false;
  33. private int dataRowID = 1;
  34. private ArrayList dataRowIDList = new ArrayList ();
  35. // this is needed for inserting new row to datatable via xml
  36. private Hashtable TempTable = new Hashtable ();
  37. DataColumnChangeEventHandler columnChanged;
  38. DataRowChangeEventHandler rowDeleted;
  39. DataRowChangeEventHandler rowChanged;
  40. CollectionChangeEventHandler tablesChanged;
  41. #endregion // Fields
  42. #region Constructors
  43. public XmlDataDocument ()
  44. {
  45. InitDelegateFields ();
  46. dataSet = new DataSet();
  47. dataSet._xmlDataDocument = this;
  48. dataSet.Tables.CollectionChanged += tablesChanged;
  49. AddXmlDocumentListeners ();
  50. DataSet.EnforceConstraints = false;
  51. }
  52. public XmlDataDocument (DataSet dataset)
  53. {
  54. if (dataset == null)
  55. throw new ArgumentException ("Parameter dataset cannot be null.");
  56. if (dataset._xmlDataDocument != null)
  57. throw new ArgumentException ("DataSet cannot be associated with two or more XmlDataDocument.");
  58. InitDelegateFields ();
  59. this.dataSet = dataset;
  60. this.dataSet._xmlDataDocument = this;
  61. XmlReader xmlReader = new XmlTextReader (new StringReader (dataSet.GetXml ()));
  62. // Load DataSet's xml-data
  63. base.Load (xmlReader);
  64. xmlReader.Close ();
  65. // FIXME: This is required to avoid Load() error when for
  66. // example empty DataSet will be filled on Load(), but
  67. // not sure if it works correct.
  68. if (DocumentElement.ChildNodes.Count == 0)
  69. RemoveChild (DocumentElement);
  70. foreach (DataTable Table in DataSet.Tables) {
  71. foreach (DataRow Row in Table.Rows) {
  72. Row.XmlRowID = dataRowID;
  73. dataRowIDList.Add (dataRowID);
  74. dataRowID++;
  75. }
  76. }
  77. AddXmlDocumentListeners ();
  78. foreach (DataTable Table in dataSet.Tables) {
  79. Table.ColumnChanged += columnChanged;
  80. Table.RowDeleted += rowDeleted;
  81. Table.RowChanged += rowChanged;
  82. }
  83. }
  84. // bool clone. If we are cloning XmlDataDocument then clone should be true.
  85. private XmlDataDocument (DataSet dataset, bool clone)
  86. {
  87. InitDelegateFields ();
  88. this.dataSet = dataset;
  89. this.dataSet._xmlDataDocument = this;
  90. foreach (DataTable Table in DataSet.Tables) {
  91. foreach (DataRow Row in Table.Rows) {
  92. Row.XmlRowID = dataRowID;
  93. dataRowIDList.Add (dataRowID);
  94. dataRowID++;
  95. }
  96. }
  97. AddXmlDocumentListeners ();
  98. foreach (DataTable Table in dataSet.Tables) {
  99. Table.ColumnChanged += columnChanged;
  100. Table.RowDeleted += rowDeleted;
  101. Table.RowChanged += rowChanged;
  102. }
  103. }
  104. #endregion // Constructors
  105. #region Public Properties
  106. public DataSet DataSet {
  107. get {
  108. return dataSet;
  109. }
  110. }
  111. #endregion // Public Properties
  112. #region Public Methods
  113. [MonoTODO]
  114. public override XmlNode CloneNode(bool deep)
  115. {
  116. XmlDataDocument Document;
  117. if (deep)
  118. Document = new XmlDataDocument (DataSet.Copy (), true);
  119. else
  120. Document = new XmlDataDocument (DataSet.Clone (), true);
  121. Document.RemoveXmlDocumentListeners ();
  122. Document.PreserveWhitespace = PreserveWhitespace;
  123. if (deep) {
  124. foreach(XmlNode n in ChildNodes)
  125. Document.AppendChild (Document.ImportNode (n, deep));
  126. }
  127. Document.AddXmlDocumentListeners ();
  128. return Document;
  129. }
  130. #region overloaded CreateElement methods
  131. public override XmlElement CreateElement(
  132. string prefix, string localName, string namespaceURI)
  133. {
  134. return base.CreateElement (prefix, localName, namespaceURI);
  135. }
  136. #endregion // overloaded CreateElement Methods
  137. // will not be supported
  138. public override XmlEntityReference CreateEntityReference(string name)
  139. {
  140. throw new NotSupportedException();
  141. }
  142. // will not be supported
  143. public override XmlElement GetElementById(string elemId)
  144. {
  145. throw new NotSupportedException();
  146. }
  147. // get the XmlElement associated with the DataRow
  148. [MonoTODO ("Exceptions")]
  149. public XmlElement GetElementFromRow(DataRow r)
  150. {
  151. if (r.XmlRowID == 0) // datarow was not in xmldatadocument
  152. throw new Exception ();
  153. int elementRow = dataRowIDList.IndexOf (r.XmlRowID);
  154. return (XmlElement)GetElementsByTagName (r.Table.TableName) [elementRow];
  155. }
  156. // get the DataRow associated with the XmlElement
  157. [MonoTODO ("Exceptions")]
  158. public DataRow GetRowFromElement(XmlElement e)
  159. {
  160. XmlElement node = e;
  161. if (node == null)
  162. return null;
  163. XPathNavigator nodeNavigator = node.CreateNavigator ();
  164. int c = GetElementsByTagName (node.Name).Count;
  165. if (c == 0)
  166. return null;
  167. XmlNodeList nodeList = GetElementsByTagName (node.Name);
  168. int i = 0;
  169. bool isSame = false;
  170. while (i < c && !isSame) {
  171. XPathNavigator docNavigator = nodeList [i].CreateNavigator ();
  172. isSame = docNavigator.IsSamePosition (nodeNavigator);
  173. docNavigator = nodeList [i].CreateNavigator ();
  174. if (!isSame)
  175. i++;
  176. }
  177. if (!isSame)
  178. return null;
  179. if (i >= dataRowIDList.Count)
  180. return null;
  181. // now we know rownum
  182. int xmlrowid = (int)dataRowIDList [i];
  183. if (xmlrowid <= 0)
  184. return null;
  185. DataTable dt = DataSet.Tables [node.Name];
  186. DataRow row = null;
  187. if (dt == null)
  188. return null;
  189. foreach (DataRow r in dt.Rows) {
  190. if (xmlrowid == r.XmlRowID) {
  191. row = r;
  192. }
  193. }
  194. return row;
  195. }
  196. #region overload Load methods
  197. public override void Load(Stream inStream) {
  198. Load (new XmlTextReader (inStream));
  199. }
  200. public override void Load(string filename) {
  201. Load (new XmlTextReader (filename));
  202. }
  203. public override void Load(TextReader txtReader) {
  204. Load (new XmlTextReader (txtReader));
  205. }
  206. public override void Load (XmlReader reader)
  207. {
  208. if (DocumentElement != null)
  209. throw new InvalidOperationException ("XmlDataDocument does not support multi-time loading. New XmlDadaDocument is always required.");
  210. bool OldEC = DataSet.EnforceConstraints;
  211. DataSet.EnforceConstraints = false;
  212. dataSet.Tables.CollectionChanged -= tablesChanged;
  213. // For reading xml to XmlDocument
  214. // XmlTextReader textReader = new XmlTextReader (
  215. // reader.BaseURI);
  216. // dont listen these events
  217. RemoveXmlDocumentListeners ();
  218. DataTable dt = null;
  219. base.Load (reader);
  220. reader = new XmlNodeReader (this);
  221. if (reader.NodeType != XmlNodeType.Element)
  222. reader.MoveToContent ();
  223. // read to next element
  224. while (reader.Read () && reader.NodeType != XmlNodeType.Element);
  225. do {
  226. // Find right table from tablecollection
  227. if (DataSet.Tables.Contains (reader.LocalName)) {
  228. dt = DataSet.Tables [reader.LocalName];
  229. // Make sure event handlers are not added twice
  230. dt.ColumnChanged -= columnChanged;
  231. dt.ColumnChanged += columnChanged;
  232. dt.RowDeleted -= rowDeleted;
  233. dt.RowDeleted += rowDeleted;
  234. dt.RowChanged -= rowChanged;
  235. dt.RowChanged += rowChanged;
  236. }
  237. else
  238. continue;
  239. // Read rows to table
  240. DataRow tempRow = dt.NewRow ();
  241. while ((reader.NodeType != XmlNodeType.EndElement ||
  242. reader.Name != dt.TableName) && reader.Read()) {
  243. switch (reader.NodeType) {
  244. case XmlNodeType.Element:
  245. // Add column to DataRow
  246. LoadRow (reader, ref tempRow);
  247. break;
  248. default:
  249. break;
  250. }
  251. }
  252. // Every row must have unique id.
  253. tempRow.XmlRowID = dataRowID;
  254. dataRowIDList.Add (dataRowID);
  255. dt.Rows.Add (tempRow);
  256. dataRowID++;
  257. } while (reader.Read ());
  258. // base.Load (textReader);
  259. // textReader.Close ();
  260. DataSet.EnforceConstraints = OldEC;
  261. AddXmlDocumentListeners ();
  262. dataSet.Tables.CollectionChanged += tablesChanged;
  263. }
  264. #endregion // overloaded Load methods
  265. #endregion // Public Methods
  266. #region Protected Methods
  267. [MonoTODO ("Create optimized XPathNavigator")]
  268. protected override XPathNavigator CreateNavigator(XmlNode node) {
  269. return base.CreateNavigator (node);
  270. }
  271. #endregion // Protected Methods
  272. #region XmlDocument event handlers
  273. private void OnNodeChanging (object sender, XmlNodeChangedEventArgs args)
  274. {
  275. if (DataSet.EnforceConstraints)
  276. throw new InvalidOperationException (Locale.GetText ("Please set DataSet.EnforceConstraints == false before trying to edit XmlDataDocument using XML operations."));
  277. }
  278. // Invoked when XmlNode is changed colum is changed
  279. [MonoTODO]
  280. private void OnNodeChanged (object sender, XmlNodeChangedEventArgs args)
  281. {
  282. if (args.Node == null)
  283. return;
  284. DataRow row = GetRowFromElement ((XmlElement)args.Node.ParentNode.ParentNode);
  285. if (row == null)
  286. return;
  287. if (!row.Table.Columns.Contains (args.Node.ParentNode.Name))
  288. return;
  289. row.Table.ColumnChanged -= columnChanged;
  290. if (row [args.Node.ParentNode.Name].ToString () != args.Node.InnerText)
  291. row [args.Node.ParentNode.Name] = args.Node.InnerText;
  292. row.Table.ColumnChanged += columnChanged;
  293. }
  294. // Invoked when XmlNode is removed
  295. [MonoTODO]
  296. private void OnNodeRemoved (object sender, XmlNodeChangedEventArgs args)
  297. {
  298. if (args.OldParent == null)
  299. return;
  300. if (!(args.OldParent is XmlElement))
  301. return;
  302. DataRow row = GetRowFromElement ((XmlElement)args.OldParent);
  303. if (row == null)
  304. return ;
  305. // Dont trig event again
  306. row.Table.ColumnChanged -= columnChanged;
  307. row [args.Node.Name] = null;
  308. row.Table.ColumnChanged += columnChanged;
  309. }
  310. private void OnNodeInserting (object sender, XmlNodeChangedEventArgs args)
  311. {
  312. if (DataSet.EnforceConstraints)
  313. throw new InvalidOperationException (Locale.GetText ("Please set DataSet.EnforceConstraints == false before trying to edit XmlDataDocument using XML operations."));
  314. }
  315. private void OnNodeInserted (object sender, XmlNodeChangedEventArgs args)
  316. {
  317. // this is table element
  318. if (DataSet.Tables.Contains (args.NewParent.Name)) {
  319. Hashtable ht = null;
  320. if (TempTable.ContainsKey (args.NewParent.Name)) {
  321. // if TempTable contains table name, get it and remove it from hashtable
  322. // so we can later add it :)
  323. ht = TempTable [args.NewParent.Name] as Hashtable;
  324. TempTable.Remove (args.NewParent.Name);
  325. }
  326. else
  327. ht = new Hashtable ();
  328. ht.Add (args.Node.Name, args.Node.InnerText);
  329. TempTable.Add (args.NewParent.Name, ht);
  330. }
  331. else if (DataSet.Tables.Contains (args.Node.Name)) {
  332. // if nodes name is same as some table in the list is is time to
  333. // add row to datatable
  334. DataTable dt = DataSet.Tables [args.Node.Name];
  335. dt.RowChanged -= rowChanged;
  336. DataRow row = dt.NewRow ();
  337. Hashtable ht = TempTable [args.Node.Name] as Hashtable;
  338. IDictionaryEnumerator enumerator = ht.GetEnumerator ();
  339. while (enumerator.MoveNext ()) {
  340. if (dt.Columns.Contains (enumerator.Key.ToString ()))
  341. row [enumerator.Key.ToString ()] = enumerator.Value.ToString ();
  342. }
  343. DataSet.Tables [args.Node.Name].Rows.Add (row);
  344. dt.RowChanged += rowChanged;
  345. }
  346. }
  347. #endregion // DataSet event handlers
  348. #region DataSet event handlers
  349. // If DataTable is added or removed from DataSet
  350. private void OnDataTableChanged (object sender, CollectionChangeEventArgs eventArgs)
  351. {
  352. DataTable Table = (DataTable)eventArgs.Element;
  353. if (eventArgs.Action == CollectionChangeAction.Add) {
  354. Table.ColumnChanged += columnChanged;
  355. Table.RowDeleted += rowDeleted;
  356. Table.RowChanged += rowChanged;
  357. }
  358. }
  359. // If column has changed
  360. [MonoTODO]
  361. private void OnDataTableColumnChanged(object sender,
  362. DataColumnChangeEventArgs eventArgs)
  363. {
  364. RemoveXmlDocumentListeners ();
  365. // row is not yet in datatable
  366. if (eventArgs.Row.XmlRowID == 0)
  367. return;
  368. // TODO: Here should be some kind of error checking.
  369. GetElementsByTagName (eventArgs.Column.ColumnName) [dataRowIDList.IndexOf (
  370. eventArgs.Row.XmlRowID)].InnerText = eventArgs.ProposedValue.ToString ();
  371. AddXmlDocumentListeners ();
  372. }
  373. [MonoTODO]
  374. private void OnDataTableRowDeleted(object sender,
  375. DataRowChangeEventArgs eventArgs)
  376. {
  377. DataRow deletedRow = null;
  378. deletedRow = eventArgs.Row;
  379. if (eventArgs.Row.XmlRowID == 0)
  380. return;
  381. int rowIndex = dataRowIDList.IndexOf (eventArgs.Row.XmlRowID);
  382. if (rowIndex == -1 || eventArgs.Row.XmlRowID == 0 ||
  383. rowIndex > GetElementsByTagName (deletedRow.Table.TableName).Count - 1)
  384. return;
  385. // Remove element from xmldocument and row indexlist
  386. // FIXME: this is one way to do this, but i hope someday i find out much better way.
  387. XmlNode p = GetElementsByTagName (deletedRow.Table.TableName) [rowIndex].ParentNode;
  388. if (p != null) {
  389. p.RemoveChild (GetElementsByTagName (deletedRow.Table.TableName) [rowIndex]);
  390. dataRowIDList.RemoveAt (rowIndex);
  391. }
  392. }
  393. [MonoTODO]
  394. private void OnDataTableRowChanged(object sender, DataRowChangeEventArgs eventArgs)
  395. {
  396. switch (eventArgs.Action) {
  397. case DataRowAction.Delete:
  398. OnDataTableRowDeleted (sender, eventArgs);
  399. break;
  400. case DataRowAction.Add:
  401. OnDataTableRowAdded (eventArgs);
  402. break;
  403. case DataRowAction.Rollback:
  404. OnDataTableRowRollback (eventArgs);
  405. break;
  406. default:
  407. break;
  408. }
  409. }
  410. // Added
  411. [MonoTODO]
  412. private void OnDataTableRowAdded (DataRowChangeEventArgs args)
  413. {
  414. RemoveXmlDocumentListeners ();
  415. // If XmlRowID is != 0 then it is already added
  416. if (args.Row.XmlRowID != 0)
  417. return;
  418. // Create row element. Row's name same as TableName
  419. DataRow row = args.Row;
  420. row.XmlRowID = dataRowID;
  421. dataRowIDList.Add (dataRowID);
  422. dataRowID++;
  423. if (DocumentElement == null)
  424. this.AppendChild (CreateElement (DataSet.DataSetName));
  425. XmlElement element = CreateElement (args.Row.Table.TableName);
  426. DocumentElement.AppendChild (element);
  427. XmlElement rowElement = null;
  428. for (int i = 0; i < row.Table.Columns.Count; i++) {
  429. rowElement = CreateElement (row.Table.Columns [i].ColumnName);
  430. object v = row [i];
  431. rowElement.InnerText = v != null ? v.ToString () : String.Empty;
  432. element.AppendChild (rowElement);
  433. }
  434. AddXmlDocumentListeners ();
  435. }
  436. // Rollback
  437. [MonoTODO]
  438. private void OnDataTableRowRollback (DataRowChangeEventArgs args)
  439. {
  440. RemoveXmlDocumentListeners ();
  441. DataRow row = args.Row;
  442. int rowid = dataRowIDList.IndexOf (row.XmlRowID);
  443. // find right element in xmldocument
  444. if (rowid == 0 || rowid >= GetElementsByTagName (row.Table.TableName).Count)
  445. return;
  446. XmlNode node = GetElementsByTagName (row.Table.TableName) [rowid];
  447. int rowValue = 0;
  448. for (int i = 0; i < node.ChildNodes.Count; i++) {
  449. XmlNode child = node.ChildNodes [i];
  450. if (child.NodeType != XmlNodeType.Whitespace) {
  451. child.InnerText = (string)row [rowValue++];
  452. }
  453. }
  454. AddXmlDocumentListeners ();
  455. }
  456. #endregion // DataSet event handlers
  457. #region Private methods
  458. private void InitDelegateFields ()
  459. {
  460. columnChanged = new DataColumnChangeEventHandler (OnDataTableColumnChanged);
  461. rowDeleted = new DataRowChangeEventHandler (OnDataTableRowDeleted);
  462. rowChanged = new DataRowChangeEventHandler (OnDataTableRowChanged);
  463. tablesChanged = new CollectionChangeEventHandler (OnDataTableChanged);
  464. }
  465. [MonoTODO]
  466. private void LoadRow (XmlReader reader, ref DataRow row)
  467. {
  468. // dt.Rows.Add (LoadRow (reader, dt.NewRow ()));
  469. // This method returns DataRow filled by values
  470. // from xmldocument
  471. string rowname = reader.Name;
  472. string column = "";
  473. if (reader.NodeType == XmlNodeType.Element)
  474. column = reader.Name;
  475. reader.Read ();
  476. if (reader.NodeType == XmlNodeType.Text) {
  477. string val = reader.Value;
  478. if (row.Table.Columns.Contains (column))
  479. row [column] = val;
  480. }
  481. }
  482. private void RemoveXmlDocumentListeners ()
  483. {
  484. this.NodeInserting -= new XmlNodeChangedEventHandler (OnNodeInserting);
  485. this.NodeInserted -= new XmlNodeChangedEventHandler (OnNodeInserted);
  486. this.NodeChanged -= new XmlNodeChangedEventHandler (OnNodeChanged);
  487. this.NodeChanging -= new XmlNodeChangedEventHandler (OnNodeChanging);
  488. this.NodeRemoved -= new XmlNodeChangedEventHandler (OnNodeRemoved);
  489. }
  490. private void AddXmlDocumentListeners ()
  491. {
  492. this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
  493. this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
  494. this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
  495. this.NodeChanging += new XmlNodeChangedEventHandler (OnNodeChanging);
  496. this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
  497. }
  498. #endregion // Private methods
  499. }
  500. }