XmlDataDocument.cs 18 KB

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