| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630 |
- //
- // 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
- //
- // 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;
- 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();
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- //this.NodeChanging += new XmlNodeChangedEventHandler (OnXmlDataColumnChanged);
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- }
- public XmlDataDocument(DataSet dataset) {
- this.dataSet = dataset;
- this.NodeChanged += new XmlNodeChangedEventHandler (OnNodeChanged);
- //this.NodeChanging += new XmlNodeChangedEventHandler (OnXmlDataColumnChanged);
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeRemoved += new XmlNodeChangedEventHandler (OnNodeRemoved);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- }
- #endregion // Constructors
- #region Public Properties
- public override string BaseURI {
- [MonoTODO]
- get {
- // TODO: why are we overriding?
- return base.BaseURI;
- }
- }
- public DataSet DataSet {
- [MonoTODO]
- get {
- return dataSet;
- }
- }
- // override inheritted method from XmlDocument
- public override string InnerXml {
- [MonoTODO]
- get {
- throw new NotImplementedException();
- }
-
- [MonoTODO]
- set {
- throw new NotImplementedException();
- }
- }
- public override bool IsReadOnly {
- [MonoTODO]
- get {
- return isReadOnly;
- }
- }
- // Item indexer
- public override XmlElement this[string name] {
- [MonoTODO]
- get {
- throw new NotImplementedException();
- }
- }
- // Item indexer
- public override XmlElement this[string localname, string ns] {
- [MonoTODO]
- get {
- throw new NotImplementedException();
- }
- }
- public override string LocalName {
- [MonoTODO]
- get {
- throw new NotImplementedException();
- }
- }
- public override string Name {
- [MonoTODO]
- get {
- return base.Name;
- }
- }
- public override XmlDocument OwnerDocument {
- [MonoTODO]
- get {
- return null;
- }
- }
- #endregion // Public Properties
- #region Public Methods
- [MonoTODO]
- public override XmlNode CloneNode(bool deep)
- {
- throw new NotImplementedException();
- }
- #region overloaded CreateElement methods
- [MonoTODO ("why this is override?")]
- public override XmlElement CreateElement(string prefix,
- string localName, string namespaceURI)
- {
- if ((localName == null) || (localName == String.Empty))
- throw new ArgumentException ("The local name for elements or attributes cannot be null" +
- "or an empty string.");
- string pref = prefix != null ? prefix : String.Empty;
- return base.CreateElement (pref, localName, namespaceURI != null ? namespaceURI : String.Empty);
- }
- #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;
- // FIXME: I dont know which way it should be but this work on linux.
- // could be also GetElementsByTagName (args.OldParent.Name) []
- 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) {
- // dont listen these events
- RemoveXmlDocumentListeners ();
- DataTable dt = null;
- // For reading xml to XmlDocument
- XmlTextReader textReader = new XmlTextReader (
- reader.BaseURI);
- if (reader.NodeType != XmlNodeType.Element)
- reader.MoveToContent ();
- // TODO: Findout which exception should be throwen
- if (reader.NodeType != XmlNodeType.Element) {
- throw new Exception ();
- }
- if (dataSet.DataSetName != reader.Name) {
- throw new Exception ();
- }
- // read to next element
- while (reader.Read () && reader.NodeType != XmlNodeType.Element);
- do {
- // Find right table from tablecollection
- for (int i = 0; i < DataSet.Tables.Count && dt == null; i++) {
- if (reader.Name == DataSet.Tables [i].TableName) {
- dt = DataSet.Tables [i];
- dt.ColumnChanged += new DataColumnChangeEventHandler (OnDataTableColumnChanged);
- dt.RowDeleted += new DataRowChangeEventHandler (OnDataTableRowDeleted);
- dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
-
- // TODO: Findout what kind of exception
- if (dt == null)
- throw new Exception (); // there were no correct table
- // 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);
- AddXmlDocumentListeners ();
- }
-
- #endregion // overloaded Load methods
- [MonoTODO]
- public override void WriteContentTo(XmlWriter xw) {
- base.WriteContentTo (xw);
- }
- [MonoTODO]
- public override void WriteTo(XmlWriter w) {
- base.WriteTo (w);
- }
- #endregion // Public Methods
- #region Protected Methods
- //FIXME: how do you handle this?
- //[MonoTODO]
- //protected internal override XPathNavigator CreateNavigator(XmlNode node) {
- // throw new NotImplementedException();
- //}
- [MonoTODO]
- public new XPathNavigator CreateNavigator() {
- throw new NotImplementedException();
- }
- #endregion // Protected Methods
-
- #region DataSet event handlers
-
- // 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);
- if (row == null)
- return;
-
- if (row [args.Node.Name].ToString () != args.Node.InnerText)
- row [args.Node.Name] = args.Node.InnerText;
- }
- // Invoked when XmlNode is removed
- [MonoTODO]
- private void OnNodeRemoved (object sender, XmlNodeChangedEventArgs args)
- {
- if (args.OldParent == null)
- 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;
- // if all columns are "nulled" we can remove the row.
- // FIXME: This is rather slow, try to find faster way
- bool allNulls = true;
- foreach (DataColumn dc in row.Table.Columns) {
-
- if (row [dc.ColumnName] != DBNull.Value) {
- allNulls = false;
- break;
- }
- }
- if (allNulls) {
- dataRowIDList.Remove (row.XmlRowID);
- row.Delete ();
- }
- 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];
- DataRow row = dt.NewRow ();
-
- Hashtable ht = TempTable [args.Node.Name] as Hashtable;
- IDictionaryEnumerator enumerator = ht.GetEnumerator ();
- while (enumerator.MoveNext ()) {
- row [enumerator.Key.ToString ()] = enumerator.Value.ToString ();
- }
-
- dt.RowChanged -= new DataRowChangeEventHandler (OnDataTableRowChanged);
- DataSet.Tables [args.Node.Name].Rows.Add (row);
- dt.RowChanged += new DataRowChangeEventHandler (OnDataTableRowChanged);
- }
- }
- [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.ToString ()) [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;
- dataRowID++;
- 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].ToString ());
- rowElement.InnerText = (string)row [i];
- 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);
- }
- private void AddXmlDocumentListeners ()
- {
- this.NodeInserting += new XmlNodeChangedEventHandler (OnNodeInserting);
- this.NodeInserted += new XmlNodeChangedEventHandler (OnNodeInserted);
- }
- #endregion // Private methods
- }
- }
|