You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			288 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			288 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="BaseDataBoundControl.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | namespace System.Web.UI.WebControls { | ||
|  | 
 | ||
|  |     using System; | ||
|  |     using System.Collections; | ||
|  |     using System.ComponentModel; | ||
|  |     using System.Web.Util; | ||
|  | 
 | ||
|  |     /// <summary> | ||
|  |     /// A BaseDataBoundControl is bound to a data source and generates its | ||
|  |     /// user interface (or child control hierarchy typically), by enumerating | ||
|  |     /// the items in the data source it is bound to. | ||
|  |     /// BaseDataBoundControl is an abstract base class that defines the common | ||
|  |     /// characteristics of all controls that use a list as a data source, such as | ||
|  |     /// DataGrid, DataBoundTable, ListBox etc. It encapsulates the logic | ||
|  |     /// of how a data-bound control binds to collections or DataControl instances. | ||
|  |     /// </summary> | ||
|  | 
 | ||
|  |     [ | ||
|  |     Designer("System.Web.UI.Design.WebControls.BaseDataBoundControlDesigner, " + AssemblyRef.SystemDesign), | ||
|  |     DefaultProperty("DataSourceID") | ||
|  |     ] | ||
|  |     public abstract class BaseDataBoundControl : WebControl { | ||
|  | 
 | ||
|  |         private static readonly object EventDataBound = new object(); | ||
|  | 
 | ||
|  |         private object _dataSource; | ||
|  |         private bool _requiresDataBinding; | ||
|  |         private bool _inited; | ||
|  |         private bool _preRendered; | ||
|  |         private bool _requiresBindToNull; | ||
|  |         private bool _throwOnDataPropertyChange; | ||
|  | 
 | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// The data source to bind to. This allows a BaseDataBoundControl to bind | ||
|  |         /// to arbitrary lists of data items. | ||
|  |         /// </summary> | ||
|  |         [ | ||
|  |         Bindable(true), | ||
|  |         DefaultValue(null), | ||
|  |         DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden), | ||
|  |         Themeable(false), | ||
|  |         WebCategory("Data"), | ||
|  |         WebSysDescription(SR.BaseDataBoundControl_DataSource), | ||
|  |         ] | ||
|  |         public virtual object DataSource { | ||
|  |             get { | ||
|  |                 return _dataSource; | ||
|  |             } | ||
|  |             set { | ||
|  |                 if (value != null) { | ||
|  |                     ValidateDataSource(value); | ||
|  |                 } | ||
|  |                 _dataSource = value; | ||
|  |                 OnDataPropertyChanged(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// The ID of the DataControl that this control should use to retrieve | ||
|  |         /// its data source. When the control is bound to a DataControl, it | ||
|  |         /// can retrieve a data source instance on-demand, and thereby attempt | ||
|  |         /// to work in auto-DataBind mode. | ||
|  |         /// </summary> | ||
|  |         [ | ||
|  |         DefaultValue(""), | ||
|  |         Themeable(false), | ||
|  |         WebCategory("Data"), | ||
|  |         WebSysDescription(SR.BaseDataBoundControl_DataSourceID) | ||
|  |         ] | ||
|  |         public virtual string DataSourceID { | ||
|  |             get { | ||
|  |                 object o = ViewState["DataSourceID"]; | ||
|  |                 if (o != null) { | ||
|  |                     return (string)o; | ||
|  |                 } | ||
|  |                 return String.Empty; | ||
|  |             } | ||
|  |             set { | ||
|  |                 if (String.IsNullOrEmpty(value) && !String.IsNullOrEmpty(DataSourceID)) { | ||
|  |                     _requiresBindToNull = true; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 ViewState["DataSourceID"] = value; | ||
|  |                 OnDataPropertyChanged(); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected bool Initialized { | ||
|  |             get { | ||
|  |                 return _inited; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Returns true if the DataBoundControl uses Select/Update/Delete/Insert Methods for databinding.  | ||
|  |         /// Implementation on BaseDataBoundControl returns false. | ||
|  |         /// Override in child classes which support the above two properties. | ||
|  |         /// </summary> | ||
|  |         protected virtual bool IsUsingModelBinders { | ||
|  |            get { | ||
|  |                return false; | ||
|  |            } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// This returns true only if the DataBoundControl is using a DataSourceID. | ||
|  |         /// Use the IsDataBindingAutomatic property to determine if the data bound control's data binding is | ||
|  |         /// automatic. The data binding is automatic if the control is using a DataSourceID or if control | ||
|  |         /// uses Select/Update/Delete/Insert Methods for data binding. | ||
|  |         /// </summary> | ||
|  |         protected bool IsBoundUsingDataSourceID { | ||
|  |             get { | ||
|  |                 return (DataSourceID.Length > 0); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// This property is used by FormView/GridView/DetailsView/ListView in the following scenarios. | ||
|  |         /// 1. Perform an auto data bind (Listen to OnDataSourceViewChanged event and set RequiresDataBinding to true) | ||
|  |         /// 2. Calling the data source view operations | ||
|  |         /// 3. Raising exceptions when there is no DataSourceId (i.e., a DataSource is in use) and an event for Data Operation is not handled | ||
|  |         /// 4. Raising the ModeChanged events for DataControls | ||
|  |         /// This property is true if the control is bound using a DataSourceId or when the control participates in Model Binding. | ||
|  |         /// </summary> | ||
|  |         protected internal bool IsDataBindingAutomatic { | ||
|  |             get { | ||
|  |                 return IsBoundUsingDataSourceID || IsUsingModelBinders; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override bool SupportsDisabledAttribute { | ||
|  |             get { | ||
|  |                 return RenderingCompatibility < VersionUtil.Framework40; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected bool RequiresDataBinding { | ||
|  |             get { | ||
|  |                 return _requiresDataBinding; | ||
|  |             } | ||
|  |             set { | ||
|  |                 // if we have to play catch-up here because we've already PreRendered, call EnsureDataBound | ||
|  |                 if (value && _preRendered && IsDataBindingAutomatic && Page != null && !Page.IsCallback) { | ||
|  |                     _requiresDataBinding = true; | ||
|  |                     EnsureDataBound(); | ||
|  |                 } | ||
|  |                 else { | ||
|  |                     _requiresDataBinding = value; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         [ | ||
|  |         WebCategory("Data"), | ||
|  |         WebSysDescription(SR.BaseDataBoundControl_OnDataBound) | ||
|  |         ] | ||
|  |         public event EventHandler DataBound { | ||
|  |             add { | ||
|  |                 Events.AddHandler(EventDataBound, value); | ||
|  |             } | ||
|  |             remove { | ||
|  |                 Events.RemoveHandler(EventDataBound, value); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected void ConfirmInitState() { | ||
|  |             _inited = true; // do this in OnLoad in case we were added to the page after Page.OnPreLoad | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         /// <summary> | ||
|  |         /// Overriden by BaseDataBoundControl to use its properties to determine the real | ||
|  |         /// data source that the control should bind to. It then clears the existing | ||
|  |         /// control hierarchy, and calls CreateChildControls to create a new control | ||
|  |         /// hierarchy based on the resolved data source. | ||
|  |         /// The implementation resolves various data source related properties to | ||
|  |         /// arrive at the appropriate IEnumerable implementation to use as the real | ||
|  |         /// data source. | ||
|  |         /// When resolving data sources, the DataSourceID takes highest precedence. | ||
|  |         /// If DataSourceID is not set, the value of the DataSource property is used. | ||
|  |         /// In this second alternative, DataMember is used to extract the appropriate | ||
|  |         /// list if the control has been handed an IListSource as a data source. | ||
|  |         /// | ||
|  |         /// Data bound controls should override PerformDataBinding instead | ||
|  |         /// of DataBind.  If DataBind if overridden, the OnDataBinding and OnDataBound events will | ||
|  |         /// fire in the wrong order.  However, for backwards compat on ListControl and AdRotator, we  | ||
|  |         /// can't seal this method.  It is sealed on all new BaseDataBoundControl-derived controls. | ||
|  |         /// </summary> | ||
|  |         public override void DataBind() { | ||
|  |             // Don't databind when the control is in the designer but not top-level | ||
|  |             if (DesignMode) { | ||
|  |                 IDictionary designModeState = GetDesignModeState(); | ||
|  |                 if (((designModeState == null) || (designModeState["EnableDesignTimeDataBinding"] == null)) | ||
|  |                     && (Site == null)) { | ||
|  |                     return; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             PerformSelect(); | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected virtual void EnsureDataBound() { | ||
|  |             try { | ||
|  |                 _throwOnDataPropertyChange = true; | ||
|  |                 if (RequiresDataBinding && (IsDataBindingAutomatic || _requiresBindToNull)) { | ||
|  |                     DataBind(); | ||
|  |                     _requiresBindToNull = false; | ||
|  |                 } | ||
|  |             } | ||
|  |             finally { | ||
|  |                 _throwOnDataPropertyChange = false; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal void InternalEnsureDataBound() { | ||
|  |             EnsureDataBound(); | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected virtual void OnDataBound(EventArgs e) { | ||
|  |             EventHandler handler = Events[EventDataBound] as EventHandler; | ||
|  |             if (handler != null) { | ||
|  |                 handler(this, e); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///  This method is called when DataMember, DataSource, or DataSourceID is changed. | ||
|  |         /// </devdoc> | ||
|  |         protected virtual void OnDataPropertyChanged() { | ||
|  |             if (_throwOnDataPropertyChange) { | ||
|  |                 throw new HttpException(SR.GetString(SR.DataBoundControl_InvalidDataPropertyChange, ID)); | ||
|  |             } | ||
|  |             if (_inited) { | ||
|  |                 RequiresDataBinding = true; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected internal override void OnInit(EventArgs e) { | ||
|  |             base.OnInit(e); | ||
|  | 
 | ||
|  |             if (Page != null) { | ||
|  |                 Page.PreLoad += new EventHandler(this.OnPagePreLoad); | ||
|  |                  | ||
|  |                 if (!IsViewStateEnabled && Page.IsPostBack) { | ||
|  |                     RequiresDataBinding = true; | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected virtual void OnPagePreLoad(object sender, EventArgs e) { | ||
|  |             _inited = true; | ||
|  |             if (Page != null) { | ||
|  |                 Page.PreLoad -= new EventHandler(this.OnPagePreLoad); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected internal override void OnPreRender(EventArgs e) { | ||
|  |             _preRendered = true; | ||
|  |             EnsureDataBound(); | ||
|  |             base.OnPreRender(e); | ||
|  |         } | ||
|  |          | ||
|  |         /// <summary> | ||
|  |         /// Override to control how the data is selected and the control is databound. | ||
|  |         /// </summary> | ||
|  |         protected abstract void PerformSelect(); | ||
|  | 
 | ||
|  | 
 | ||
|  |         protected abstract void ValidateDataSource(object dataSource); | ||
|  |     } | ||
|  | } | ||
|  | 
 |