2 // System.Web.UI.WebControls.BaseDataBoundControl.cs
5 // Lluis Sanchez Gual (lluis@novell.com)
7 // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com)
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 using System.Collections;
31 using System.ComponentModel;
32 using System.Security.Permissions;
34 namespace System.Web.UI.WebControls {
37 [AspNetHostingPermission (SecurityAction.LinkDemand, Level = AspNetHostingPermissionLevel.Minimal)]
38 [AspNetHostingPermission (SecurityAction.InheritanceDemand, Level = AspNetHostingPermissionLevel.Minimal)]
40 [DefaultProperty ("DataSourceID")]
41 [DesignerAttribute ("System.Web.UI.Design.WebControls.BaseDataBoundControlDesigner, " + Consts.AssemblySystem_Design, "System.ComponentModel.Design.IDesigner")]
42 public abstract class BaseDataBoundControl: WebControl
44 static readonly object dataBoundEvent = new object ();
46 EventHandlerList events = new EventHandlerList ();
48 public event EventHandler DataBound {
49 add { events.AddHandler (dataBoundEvent, value); }
50 remove { events.RemoveHandler (dataBoundEvent, value); }
56 bool requiresDataBinding;
58 protected BaseDataBoundControl ()
62 /* Used for controls that used to inherit from
63 * WebControl, so the tag can propagate upwards
65 internal BaseDataBoundControl (HtmlTextWriterTag tag) : base (tag)
69 [BindableAttribute (true)]
70 [ThemeableAttribute (false)]
71 [DefaultValueAttribute (null)]
72 [DesignerSerializationVisibilityAttribute (DesignerSerializationVisibility.Hidden)]
73 public virtual object DataSource {
79 ValidateDataSource (value);
81 OnDataPropertyChanged ();
85 [DefaultValueAttribute ("")]
86 [ThemeableAttribute (false)]
87 public virtual string DataSourceID {
89 return ViewState.GetString ("DataSourceID", "");
92 ViewState["DataSourceID"] = value;
93 OnDataPropertyChanged ();
97 protected bool Initialized {
98 get { return initialized; }
101 protected bool IsBoundUsingDataSourceID {
102 get { return DataSourceID.Length > 0; }
105 protected bool RequiresDataBinding {
106 get { return requiresDataBinding; }
108 // MSDN: If you set the RequiresDataBinding
109 // property to true when the data-bound control
110 // has already begun to render its output to the
111 // page, the current HTTP request is not a
112 // callback, and you are using the DataSourceID
113 // property to identify the data source control
114 // to bind to, the DataBind method is called
115 // immediately. In this case, the
116 // RequiresDataBinding property is _not_ actually
119 // LAMESPEC, the docs quoted above mention that
120 // DataBind is called in the described
121 // case. This is wrong since that way we don't
122 // break recursion when the property is set from
123 // within the OnSelect handler in user's
124 // code. EnsureDataBound makes sure that no
125 // recursive binding is performed. Also the
126 // property DOES get set in this case (according
128 if (value && preRendered && IsBoundUsingDataSourceID && Page != null && !Page.IsCallback) {
129 requiresDataBinding = true;
132 requiresDataBinding = value;
136 protected void ConfirmInitState ()
141 public override void DataBind ()
146 protected virtual void EnsureDataBound ()
148 if (RequiresDataBinding && IsBoundUsingDataSourceID)
152 protected virtual void OnDataBound (EventArgs e)
154 EventHandler eh = events [dataBoundEvent] as EventHandler;
159 protected virtual void OnDataPropertyChanged ()
162 RequiresDataBinding = true;
165 protected internal override void OnInit (EventArgs e)
168 Page.PreLoad += new EventHandler (OnPagePreLoad);
170 if (!IsViewStateEnabled && Page != null && Page.IsPostBack)
171 RequiresDataBinding = true;
174 protected virtual void OnPagePreLoad (object sender, EventArgs e)
179 protected internal override void OnPreRender (EventArgs e)
183 base.OnPreRender (e);
186 internal Control FindDataSource ()
189 Control namingContainer = NamingContainer;
191 while (namingContainer != null) {
192 ctrl = namingContainer.FindControl (DataSourceID);
195 namingContainer = namingContainer.NamingContainer;
201 protected abstract void PerformSelect ();
203 protected abstract void ValidateDataSource (object dataSource);