//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // // [....] // [....] //------------------------------------------------------------------------------ namespace System.Data { using System.Diagnostics; using System.ComponentModel; public class DataRowView : ICustomTypeDescriptor, System.ComponentModel.IEditableObject, System.ComponentModel.IDataErrorInfo, System.ComponentModel.INotifyPropertyChanged { private readonly DataView dataView; private readonly DataRow _row; private bool delayBeginEdit; private static PropertyDescriptorCollection zeroPropertyDescriptorCollection = new PropertyDescriptorCollection(null); /// /// When the PropertyChanged event happens, it must happen on the same DataRowView reference. /// This is so a generic event handler like Windows Presentation Foundation can redirect as appropriate. /// Having DataView.Equals is not sufficient for WPF, because two different instances may be equal but not equivalent. /// For DataRowView, if two instances are equal then they are equivalent. /// private System.ComponentModel.PropertyChangedEventHandler onPropertyChanged; internal DataRowView(DataView dataView, DataRow row) { this.dataView = dataView; this._row = row; } /// /// Checks for same reference instead of equivalent or . /// /// Necessary for ListChanged event handlers to use data structures that use the default to /// instead of /// to understand if they need to add a event handler. /// /// public override bool Equals(object other) { return Object.ReferenceEquals(this, other); } /// Hashcode of public override Int32 GetHashCode() { // Everett compatability, must return hashcode for DataRow // this does prevent using this object in collections like Hashtable // which use the hashcode as an immutable value to identify this object // user could/should have used the DataRow property instead of the hashcode return Row.GetHashCode(); } public DataView DataView { get { return dataView; } } internal int ObjectID { get { return _row.ObjectID; } } /// Gets or sets a value in specified column. /// Specified column index. /// Uses either or to access /// when setting a value. /// public object this[int ndx] { get { return Row[ndx, RowVersionDefault]; } set { if (!dataView.AllowEdit && !IsNew) { throw ExceptionBuilder.CanNotEdit(); } SetColumnValue(dataView.Table.Columns[ndx], value); } } /// Gets the specified column value or related child view or sets a value in specified column. /// Specified column or relation name when getting. Specified column name when setting. /// when is ambigous. /// Unmatched when getting a value. /// Unmatched when setting a value. /// when setting a value. public object this[string property] { get { DataColumn column = dataView.Table.Columns[property]; if (null != column) { return Row[column, RowVersionDefault]; } else if (dataView.Table.DataSet != null && dataView.Table.DataSet.Relations.Contains(property)) { return CreateChildView(property); } throw ExceptionBuilder.PropertyNotFound(property, dataView.Table.TableName); } set { DataColumn column = dataView.Table.Columns[property]; if (null == column) { throw ExceptionBuilder.SetFailed(property); } if (!dataView.AllowEdit && !IsNew) { throw ExceptionBuilder.CanNotEdit(); } SetColumnValue(column, value); } } // IDataErrorInfo stuff string System.ComponentModel.IDataErrorInfo.this[string colName] { get { return Row.GetColumnError(colName); } } string System.ComponentModel.IDataErrorInfo.Error { get { return Row.RowError; } } /// /// Gets the current version description of the /// in relation to /// /// Either or public DataRowVersion RowVersion { get { return (RowVersionDefault & ~DataRowVersion.Proposed); } } /// Either or private DataRowVersion RowVersionDefault { get { return Row.GetDefaultRowVersion(dataView.RowStateFilter); } } internal int GetRecord() { return Row.GetRecordFromVersion(RowVersionDefault); } internal bool HasRecord() { return Row.HasVersion(RowVersionDefault); } internal object GetColumnValue(DataColumn column) { return Row[column, RowVersionDefault]; } internal void SetColumnValue(DataColumn column, object value) { if (delayBeginEdit) { delayBeginEdit = false; Row.BeginEdit(); } if (DataRowVersion.Original == RowVersionDefault) { throw ExceptionBuilder.SetFailed(column.ColumnName); } Row[column] = value; } /// /// Returns a /// for the child /// with the specified . /// /// Specified . /// null or mismatch between and . public DataView CreateChildView(DataRelation relation, bool followParent) { if (relation == null || relation.ParentKey.Table != DataView.Table) { throw ExceptionBuilder.CreateChildView(); } RelatedView childView; if (!followParent) { int record = GetRecord(); object[] values = relation.ParentKey.GetKeyValues(record); childView = new RelatedView(relation.ChildColumnsReference, values); } else { childView = new RelatedView(this, relation.ParentKey, relation.ChildColumnsReference); } childView.SetIndex("", DataViewRowState.CurrentRows, null); // finish construction via RelatedView.SetIndex childView.SetDataViewManager(DataView.DataViewManager); return childView; } public DataView CreateChildView(DataRelation relation) { return CreateChildView(relation, followParent: false); } /// /// Specified name. /// Unmatched . public DataView CreateChildView(string relationName, bool followParent) { return CreateChildView(DataView.Table.ChildRelations[relationName], followParent); } public DataView CreateChildView(string relationName) { return CreateChildView(relationName, followParent: false); } public DataRow Row { get { return _row; } } public void BeginEdit() { delayBeginEdit = true; } public void CancelEdit() { DataRow tmpRow = Row; if (IsNew) { dataView.FinishAddNew(false); } else { tmpRow.CancelEdit(); } delayBeginEdit = false; } public void EndEdit() { if (IsNew) { dataView.FinishAddNew(true); } else { Row.EndEdit(); } delayBeginEdit = false; } public bool IsNew { get { return (_row == dataView.addNewRow); } } public bool IsEdit { get { return ( Row.HasVersion(DataRowVersion.Proposed) || // It was edited or delayBeginEdit); // DataRowView.BegingEdit() was called, but not edited yet. } } public void Delete() { dataView.Delete(Row); } #region ICustomTypeDescriptor AttributeCollection ICustomTypeDescriptor.GetAttributes() { return new AttributeCollection((Attribute[])null); } string ICustomTypeDescriptor.GetClassName() { return null; } string ICustomTypeDescriptor.GetComponentName() { return null; } TypeConverter ICustomTypeDescriptor.GetConverter() { return null; } EventDescriptor ICustomTypeDescriptor.GetDefaultEvent() { return null; } PropertyDescriptor ICustomTypeDescriptor.GetDefaultProperty() { return null; } object ICustomTypeDescriptor.GetEditor(Type editorBaseType) { return null; } EventDescriptorCollection ICustomTypeDescriptor.GetEvents() { return new EventDescriptorCollection(null); } EventDescriptorCollection ICustomTypeDescriptor.GetEvents(Attribute[] attributes) { return new EventDescriptorCollection(null); } PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties() { return((ICustomTypeDescriptor)this).GetProperties(null); } PropertyDescriptorCollection ICustomTypeDescriptor.GetProperties(Attribute[] attributes) { return (dataView.Table != null ? dataView.Table.GetPropertyDescriptorCollection(attributes) : zeroPropertyDescriptorCollection); } object ICustomTypeDescriptor.GetPropertyOwner(PropertyDescriptor pd) { return this; } #endregion public event PropertyChangedEventHandler PropertyChanged { add { onPropertyChanged += value; } remove { onPropertyChanged -= value; } } internal void RaisePropertyChangedEvent (string propName){ // Do not try catch, we would mask users bugs. if they throw we would catch if (onPropertyChanged != null) { onPropertyChanged (this, new PropertyChangedEventArgs(propName)); } } } }