| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638 |
- //
- // mcs/class/System.Data/System.Xml/XmlDataDocument.cs
- //
- // Purpose: Provides a W3C XML DOM Document to interact with
- // relational data in a DataSet
- //
- // class: XmlDataDocument
- // assembly: System.Data.dll
- // namespace: System.Xml
- //
- // Author:
- // Daniel Morgan <[email protected]>
- // Ville Palo <[email protected]>
- //
- // (c)copyright 2002 Daniel Morgan
- // (c)copyright 2003 Ville Palo
- //
- // XmlDataDocument is included within the Mono Class Library.
- //
- using System;
- using System.Data;
- using System.IO;
- using System.Text;
- using System.Xml.XPath;
- using System.Collections;
- using System.Globalization;
- using System.ComponentModel;
- namespace System.Xml {
- public class XmlDataDocument : XmlDocument {
- #region Fields
- private DataSet dataSet;
- private bool isReadOnly = false;
- private int dataRowID = 1;
- private ArrayList dataRowIDList = new ArrayList ();
- // this is needed for inserting new row to datatable via xml
- private Hashtable TempTable = new Hashtable ();
- #endregion // Fields
- #region Constructors
- public XmlDataDocument() {
- dataSet = new DataSet();
- dataSet.Tables.CollectionChanged += new CollectionChangeEventHandler (OnDataTableChanged);
-
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- this.NodeChanging += new XmlNodeChangedEventHandler (OnNodeChanging);
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- DataSet.EnforceConstraints = false;
- }
- public XmlDataDocument(DataSet dataset) {
- this.dataSet = dataset;
- XmlReader xmlReader = new XmlTextReader (new StringReader (dataSet.GetXml ()));
- // Load DataSet's xml-data
- base.Load (xmlReader);
- xmlReader.Close ();
- foreach (DataTable Table in DataSet.Tables) {
-
- foreach (DataRow Row in Table.Rows) {
- Row.XmlRowID = dataRowID;
- dataRowIDList.Add (dataRowID);
- dataRowID++;
- }
- }
-
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- this.NodeChanging += new XmlNodeChangedEventHandler (OnNodeChanging);
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- foreach (DataTable Table in dataSet.Tables) {
- Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- Table.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
- Table.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
- // bool clone. If we are cloning XmlDataDocument then clone should be true.
- private XmlDataDocument (DataSet dataset, bool clone)
- {
- this.dataSet = dataset;
- foreach (DataTable Table in DataSet.Tables) {
-
- foreach (DataRow Row in Table.Rows) {
- Row.XmlRowID = dataRowID;
- dataRowIDList.Add (dataRowID);
- dataRowID++;
- }
- }
-
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- this.NodeChanging += new XmlNodeChangedEventHandler (OnNodeChanging);
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- foreach (DataTable Table in dataSet.Tables) {
- Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- Table.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
- Table.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
- #endregion // Constructors
- #region Public Properties
- public DataSet DataSet {
- get {
- return dataSet;
- }
- }
- #endregion // Public Properties
- #region Public Methods
- [MonoTODO]
- public override XmlNode CloneNode(bool deep)
- {
- XmlDataDocument Document;
- if (deep)
- Document = new XmlDataDocument (DataSet.Copy (), true);
- else
- Document = new XmlDataDocument (DataSet.Clone (), true);
- RemoveXmlDocumentListeners ();
- Document.PreserveWhitespace = PreserveWhitespace;
- if (deep) {
- foreach(XmlNode n in ChildNodes)
- Document.AppendChild (Document.ImportNode (n, deep));
- }
- AddXmlDocumentListeners ();
- return Document;
- }
- #region overloaded CreateElement methods
- public override XmlElement CreateElement(
- string prefix, string localName, string namespaceURI)
- {
- return base.CreateElement (prefix, localName, namespaceURI);
- }
- #endregion // overloaded CreateElement Methods
-
- // will not be supported
- public override XmlEntityReference CreateEntityReference(string name)
- {
- throw new NotSupportedException();
- }
-
- // will not be supported
- public override XmlElement GetElementById(string elemId)
- {
- throw new NotSupportedException();
- }
- // get the XmlElement associated with the DataRow
- [MonoTODO ("Exceptions")]
- public XmlElement GetElementFromRow(DataRow r)
- {
- if (r.XmlRowID == 0) // datarow was not in xmldatadocument
- throw new Exception ();
- int elementRow = dataRowIDList.IndexOf (r.XmlRowID);
-
- return (XmlElement)GetElementsByTagName (r.Table.TableName) [elementRow];
- }
- // get the DataRow associated with the XmlElement
- [MonoTODO ("Exceptions")]
- public DataRow GetRowFromElement(XmlElement e)
- {
- XmlElement node = e;
- if (node == null)
- return null;
- XPathNavigator nodeNavigator = node.CreateNavigator ();
- int c = GetElementsByTagName (node.Name).Count;
-
- if (c == 0)
- return null;
- XmlNodeList nodeList = GetElementsByTagName (node.Name);
- int i = 0;
- bool isSame = false;
- while (i < c && !isSame) {
- XPathNavigator docNavigator = nodeList [i].CreateNavigator ();
- isSame = docNavigator.IsSamePosition (nodeNavigator);
- docNavigator = nodeList [i].CreateNavigator ();
- if (!isSame)
- i++;
- }
- if (!isSame)
- return null;
- if (i >= dataRowIDList.Count)
- return null;
- // now we know rownum
- int xmlrowid = (int)dataRowIDList [i];
- if (xmlrowid <= 0)
- return null;
- DataTable dt = DataSet.Tables [node.Name];
- DataRow row = null;
- if (dt == null)
- return null;
- foreach (DataRow r in dt.Rows) {
- if (xmlrowid == r.XmlRowID) {
- row = r;
- }
- }
- return row;
- }
- #region overload Load methods
- public override void Load(Stream inStream) {
- Load (new XmlTextReader (inStream));
- }
- public override void Load(string filename) {
- Load (new XmlTextReader (filename));
- }
- public override void Load(TextReader txtReader) {
- Load (new XmlTextReader (txtReader));
- }
- public override void Load(XmlReader reader) {
- bool OldEC = DataSet.EnforceConstraints;
- DataSet.EnforceConstraints = false;
- dataSet.Tables.CollectionChanged -= new CollectionChangeEventHandler (OnDataTableChanged);
- // For reading xml to XmlDocument
- XmlTextReader textReader = new XmlTextReader (
- reader.BaseURI);
-
- // dont listen these events
- RemoveXmlDocumentListeners ();
- DataTable dt = null;
- if (reader.NodeType != XmlNodeType.Element)
- reader.MoveToContent ();
- // read to next element
- while (reader.Read () && reader.NodeType != XmlNodeType.Element);
- do {
- // Find right table from tablecollection
- if (DataSet.Tables.Contains (reader.LocalName)) {
- dt = DataSet.Tables [reader.LocalName];
- // Make sure event handlers are not added twice
- dt.ColumnChanged -= new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- dt.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- dt.RowDeleted -= new DataRowChangeEventHandler (OnDataTableRowDeleted);
- dt.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
-
- dt.RowChanged -= new DataRowChangeEventHandler (OnDataTableRowChanged);
- dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- else
- continue;
- // Read rows to table
- DataRow tempRow = dt.NewRow ();
- while ((reader.NodeType != XmlNodeType.EndElement ||
- reader.Name != dt.TableName) && reader.Read()) {
-
- switch (reader.NodeType) {
-
- case XmlNodeType.Element:
- // Add column to DataRow
- LoadRow (reader, ref tempRow);
- break;
- default:
- break;
- }
- }
- // Every row must have unique id.
- tempRow.XmlRowID = dataRowID;
- dataRowIDList.Add (dataRowID);
- dt.Rows.Add (tempRow);
- dataRowID++;
-
-
- } while (reader.Read ());
- base.Load (textReader);
- textReader.Close ();
- DataSet.EnforceConstraints = OldEC;
- AddXmlDocumentListeners ();
- dataSet.Tables.CollectionChanged += new CollectionChangeEventHandler (OnDataTableChanged);
- }
-
- #endregion // overloaded Load methods
- #endregion // Public Methods
- #region Protected Methods
- //FIXME: when internal protected bug is fixed uncomment this
- //[MonoTODO]
- //protected internal override XPathNavigator CreateNavigator(XmlNode node) {
- // throw new NotImplementedException();
- //}
- #endregion // Protected Methods
-
- #region XmlDocument event handlers
- private void OnNodeChanging (object sender, XmlNodeChangedEventArgs args)
- {
- if (DataSet.EnforceConstraints)
- throw new InvalidOperationException (Locale.GetText ("Please set DataSet.EnforceConstraints == false " +
- "before trying to edit XmlDataDocument using " +
- "XML operations."));
- }
- // Invoked when XmlNode is changed colum is changed
- [MonoTODO]
- private void OnNodeChanged (object sender, XmlNodeChangedEventArgs args)
- {
- if (args.Node == null)
- return;
- DataRow row = GetRowFromElement ((XmlElement)args.Node.ParentNode.ParentNode);
- if (row == null)
- return;
- if (!row.Table.Columns.Contains (args.Node.ParentNode.Name))
- return;
- row.Table.ColumnChanged -= new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- if (row [args.Node.ParentNode.Name].ToString () != args.Node.InnerText)
- row [args.Node.ParentNode.Name] = args.Node.InnerText;
- row.Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- }
- // Invoked when XmlNode is removed
- [MonoTODO]
- private void OnNodeRemoved (object sender, XmlNodeChangedEventArgs args)
- {
- if (args.OldParent == null)
- return;
- if (!(args.OldParent is XmlElement))
- return;
-
- DataRow row = GetRowFromElement ((XmlElement)args.OldParent);
-
- if (row == null)
- return ;
- // Dont trig event again
- row.Table.ColumnChanged -= new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- row [args.Node.Name] = null;
- row.Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- }
- private void OnNodeInserting (object sender, XmlNodeChangedEventArgs args)
- {
- if (DataSet.EnforceConstraints)
- throw new InvalidOperationException (Locale.GetText ("Please set DataSet.EnforceConstraints == false " +
- "before trying to edit XmlDataDocument using " +
- "XML operations."));
-
- }
-
- private void OnNodeInserted (object sender, XmlNodeChangedEventArgs args)
- {
- // this is table element
- if (DataSet.Tables.Contains (args.NewParent.Name)) {
- Hashtable ht = null;
- if (TempTable.ContainsKey (args.NewParent.Name)) {
- // if TempTable contains table name, get it and remove it from hashtable
- // so we can later add it :)
- ht = TempTable [args.NewParent.Name] as Hashtable;
- TempTable.Remove (args.NewParent.Name);
- }
- else
- ht = new Hashtable ();
- ht.Add (args.Node.Name, args.Node.InnerText);
- TempTable.Add (args.NewParent.Name, ht);
- }
- else if (DataSet.Tables.Contains (args.Node.Name)) {
-
- // if nodes name is same as some table in the list is is time to
- // add row to datatable
- DataTable dt = DataSet.Tables [args.Node.Name];
- dt.RowChanged -= new DataRowChangeEventHandler (OnDataTableRowChanged);
- DataRow row = dt.NewRow ();
- Hashtable ht = TempTable [args.Node.Name] as Hashtable;
-
- IDictionaryEnumerator enumerator = ht.GetEnumerator ();
- while (enumerator.MoveNext ()) {
- if (dt.Columns.Contains (enumerator.Key.ToString ()))
- row [enumerator.Key.ToString ()] = enumerator.Value.ToString ();
- }
-
- DataSet.Tables [args.Node.Name].Rows.Add (row);
- dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
- #endregion // DataSet event handlers
- #region DataSet event handlers
- // If DataTable is added or removed from DataSet
- private void OnDataTableChanged (object sender, CollectionChangeEventArgs eventArgs)
- {
- DataTable Table = (DataTable)eventArgs.Element;
- if (eventArgs.Action == CollectionChangeAction.Add) {
- Table.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- Table.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
- Table.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
- // If column has changed
- [MonoTODO]
- private void OnDataTableColumnChanged(object sender,
- DataColumnChangeEventArgs eventArgs)
- {
- RemoveXmlDocumentListeners ();
- // row is not yet in datatable
- if (eventArgs.Row.XmlRowID == 0)
- return;
- // TODO: Here should be some kind of error checking.
- GetElementsByTagName (eventArgs.Column.ColumnName) [dataRowIDList.IndexOf (
- eventArgs.Row.XmlRowID)].InnerText = eventArgs.ProposedValue.ToString ();
-
- AddXmlDocumentListeners ();
- }
-
- [MonoTODO]
- private void OnDataTableRowDeleted(object sender,
- DataRowChangeEventArgs eventArgs)
- {
- DataRow deletedRow = null;
- deletedRow = eventArgs.Row;
- if (eventArgs.Row.XmlRowID == 0)
- return;
-
- int rowIndex = dataRowIDList.IndexOf (eventArgs.Row.XmlRowID);
- if (rowIndex == -1 || eventArgs.Row.XmlRowID == 0 ||
- rowIndex > GetElementsByTagName (deletedRow.Table.TableName).Count - 1)
- return;
-
- // Remove element from xmldocument and row indexlist
- // FIXME: this is one way to do this, but i hope someday i find out much better way.
- XmlNode p = GetElementsByTagName (deletedRow.Table.TableName) [rowIndex].ParentNode;
- if (p != null) {
- p.RemoveChild (GetElementsByTagName (deletedRow.Table.TableName) [rowIndex]);
- dataRowIDList.RemoveAt (rowIndex);
- }
- }
-
- [MonoTODO]
- private void OnDataTableRowChanged(object sender, DataRowChangeEventArgs eventArgs)
- {
- switch (eventArgs.Action) {
- case DataRowAction.Delete:
- OnDataTableRowDeleted (sender, eventArgs);
- break;
- case DataRowAction.Add:
- OnDataTableRowAdded (eventArgs);
- break;
- case DataRowAction.Rollback:
- OnDataTableRowRollback (eventArgs);
- break;
- default:
- break;
- }
- }
- // Added
- [MonoTODO]
- private void OnDataTableRowAdded (DataRowChangeEventArgs args)
- {
- RemoveXmlDocumentListeners ();
- // If XmlRowID is != 0 then it is already added
- if (args.Row.XmlRowID != 0)
- return;
-
- // Create row element. Row's name same as TableName
- DataRow row = args.Row;
- row.XmlRowID = dataRowID;
- dataRowIDList.Add (dataRowID);
- dataRowID++;
- if (DocumentElement == null)
- this.AppendChild (CreateElement (DataSet.DataSetName));
- XmlElement element = CreateElement (args.Row.Table.TableName);
- DocumentElement.AppendChild (element);
- XmlElement rowElement = null;
- for (int i = 0; i < row.Table.Columns.Count; i++) {
- rowElement = CreateElement (row.Table.Columns [i].ColumnName);
- object v = row [i];
- rowElement.InnerText = v != null ? v.ToString () : String.Empty;
- element.AppendChild (rowElement);
- }
-
- AddXmlDocumentListeners ();
- }
- // Rollback
- [MonoTODO]
- private void OnDataTableRowRollback (DataRowChangeEventArgs args)
- {
- RemoveXmlDocumentListeners ();
- DataRow row = args.Row;
- int rowid = dataRowIDList.IndexOf (row.XmlRowID);
- // find right element in xmldocument
- if (rowid == 0 || rowid >= GetElementsByTagName (row.Table.TableName).Count)
- return;
- XmlNode node = GetElementsByTagName (row.Table.TableName) [rowid];
-
- int rowValue = 0;
- for (int i = 0; i < node.ChildNodes.Count; i++) {
-
- XmlNode child = node.ChildNodes [i];
- if (child.NodeType != XmlNodeType.Whitespace) {
- child.InnerText = (string)row [rowValue++];
- }
- }
- AddXmlDocumentListeners ();
- }
- #endregion // DataSet event handlers
- #region Private methods
- [MonoTODO]
- private void LoadRow (XmlReader reader, ref DataRow row)
- {
- // dt.Rows.Add (LoadRow (reader, dt.NewRow ()));
- // This method returns DataRow filled by values
- // from xmldocument
- string rowname = reader.Name;
- string column = "";
-
- if (reader.NodeType == XmlNodeType.Element)
- column = reader.Name;
-
- reader.Read ();
-
- if (reader.NodeType == XmlNodeType.Text) {
-
- string val = reader.Value;
- if (row.Table.Columns.Contains (column))
- row [column] = val;
- }
- }
-
- private void RemoveXmlDocumentListeners ()
- {
- this.NodeInserting -= new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeInserted -= new XmlNodeChangedEventHandler (OnNodeInserted);
- this.NodeChanged -= new XmlNodeChangedEventHandler (OnNodeChanged);
- this.NodeChanging -= new XmlNodeChangedEventHandler (OnNodeChanging);
- }
- private void AddXmlDocumentListeners ()
- {
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- this.NodeChanging += new XmlNodeChangedEventHandler (OnNodeChanging);
- }
- #endregion // Private methods
- }
- }
|