我曾经一直梦想拥有象windows窗体开发中一样易使用的WebMenu、WebTreeNavigatorPanel、WebDropDownTree等树结构的Web控件,当理解asp.net后我发现实现他们将不再是一个梦,目前我已经初步开发出了这些树状控件,在今后有时间的日子愿把开发心得与大家分享。

一、树状结构框架
首先针对视图状态我开发了两个基类ViewStatePartBase和ViewStatePartCollectionBase,今后所有Menu、TreeNavigator、DropDownTree的Item、ItemColleciton均从他们继承。为了便于理解这两个基类的作用,首先我们先来开发一个简单的控件WinPop来说明基类的用法。

WinPop组件主要用在网站的首页,当用户请求主页时,它会根据控件中的Items属性(弹出窗口链接地址,弹出窗口属性等等)来弹出多个窗口。其开发原型如下: 
论Web控件开发 - 树状控件(一)

以下是ViewStatePartBase的源代码

论Web控件开发 - 树状控件(一)using System;
论Web控件开发 - 树状控件(一)
using System.Web.UI;
论Web控件开发 - 树状控件(一)
using System.Web.UI.WebControls;
论Web控件开发 - 树状控件(一)
using System.Web.UI.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing;
论Web控件开发 - 树状控件(一)
using System.ComponentModel;
论Web控件开发 - 树状控件(一)
using System.ComponentModel.Design;
论Web控件开发 - 树状控件(一)
using System.Collections;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
namespace Keyss.WebControls
{
论Web控件开发 - 树状控件(一)    
public abstract class ViewStatePartBase:IStateManager
{

论Web控件开发 - 树状控件(一)        
protected bool _isTrackingViewState;
论Web控件开发 - 树状控件(一)        
protected StateBag _viewState;
论Web控件开发 - 树状控件(一) 
论Web控件开发 - 树状控件(一)        [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        [Browsable(
false)]
论Web控件开发 - 树状控件(一)        
public virtual bool IsTrackingViewState 
{
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void LoadViewState(object savedState) 
{
论Web控件开发 - 树状控件(一)            
if(savedState != null)
{
论Web控件开发 - 树状控件(一)                ((IStateManager)ViewState).LoadViewState(savedState);
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual object SaveViewState() 
{
论Web控件开发 - 树状控件(一)            
object savedState = null;
论Web控件开发 - 树状控件(一)            
if(!IsEmpty) 
论Web控件开发 - 树状控件(一)                savedState 
= ((IStateManager)_viewState).SaveViewState();
论Web控件开发 - 树状控件(一)            
return savedState;
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void TrackViewState() 
{
论Web控件开发 - 树状控件(一)            _isTrackingViewState 
= true;
论Web控件开发 - 树状控件(一)            
if(_viewState!=null)
论Web控件开发 - 树状控件(一)                ((IStateManager)_viewState).TrackViewState();
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        Browsable(
false),
论Web控件开发 - 树状控件(一)        DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
protected virtual StateBag ViewState 
{
论Web控件开发 - 树状控件(一)            
get 
{
论Web控件开发 - 树状控件(一)                
if(_viewState == null
{
论Web控件开发 - 树状控件(一)                    _viewState 
= new StateBag(false);
论Web控件开发 - 树状控件(一)                    
if(_isTrackingViewState) 
{
论Web控件开发 - 树状控件(一)                        ((IStateManager)_viewState).TrackViewState();
论Web控件开发 - 树状控件(一)                    }

论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                
return _viewState;
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
论Web控件开发 - 树状控件(一)        [Browsable(
false)]
论Web控件开发 - 树状控件(一)        
public virtual bool IsEmpty 
{
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void SetDirty()
{
论Web控件开发 - 树状控件(一)            
if(!IsEmpty)
{
论Web控件开发 - 树状控件(一)                ICollection Keys 
= _viewState.Keys;
论Web控件开发 - 树状控件(一)                
foreach(string key in Keys)
{
论Web控件开发 - 树状控件(一)                    _viewState.SetItemDirty(key, 
true);
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual bool IsSet(string key)
{
论Web控件开发 - 树状控件(一)            
if((!IsEmpty)&&(_viewState[key]!=null))
论Web控件开发 - 树状控件(一)                
return true;
论Web控件开发 - 树状控件(一)            
return false;
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void RemoveItem(string key)
{
论Web控件开发 - 树状控件(一)            
if((_viewState[key]!=null))
论Web控件开发 - 树状控件(一)                
this._viewState.Remove(key);
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void Reset()
{
论Web控件开发 - 树状控件(一)            _viewState 
= null;
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
protected virtual void CopyFrom(ViewStatePartBase s)
{    
论Web控件开发 - 树状控件(一)            
if((s!=null)&&(!s.IsEmpty))
{    
论Web控件开发 - 树状控件(一)                
foreach(string key in s.ViewState.Keys)
{
论Web控件开发 - 树状控件(一)                    ViewState.Add(key,s.ViewState[key]);
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
protected virtual void MergeWith(ViewStatePartBase s)
{
论Web控件开发 - 树状控件(一)            
if((s!=null)&&(!s.IsEmpty))
{
论Web控件开发 - 树状控件(一)                
foreach(string key in s.ViewState.Keys)
{
论Web控件开发 - 树状控件(一)                    
if(!IsSet(key))
论Web控件开发 - 树状控件(一)                        ViewState.Add(key,s.ViewState[key]);
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion
论Web控件开发 - 树状控件(一)    }

论Web控件开发 - 树状控件(一)}

论Web控件开发 - 树状控件(一)


从上面的源代码可以看到ViewStatePartBase实现了IStateManager接口,除了标准的四个接口函数以外还实现了几个工具函数用来帮助对视图状态中保存的条目进行操作。其中Reset、MergeWith和CopyFrom主要针对自定义的样式类(如MenuItemStyle等等)操作。

以下是ViewStatePartBaseCollection的源代码

论Web控件开发 - 树状控件(一)using System;
论Web控件开发 - 树状控件(一)
using System.Web.UI;
论Web控件开发 - 树状控件(一)
using System.Web.UI.WebControls;
论Web控件开发 - 树状控件(一)
using System.Web.UI.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing;
论Web控件开发 - 树状控件(一)
using System.Collections;
论Web控件开发 - 树状控件(一)
using System.ComponentModel;
论Web控件开发 - 树状控件(一)
using System.ComponentModel.Design;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
namespace Keyss.WebControls
{
论Web控件开发 - 树状控件(一)    [ 
论Web控件开发 - 树状控件(一)    PersistenceMode(PersistenceMode.InnerProperty),
论Web控件开发 - 树状控件(一)    ] 
论Web控件开发 - 树状控件(一)    
public abstract class ViewStatePartCollectionBase:System.Collections.CollectionBase,IStateManager
{

论Web控件开发 - 树状控件(一)        
protected bool _isTrackingViewState;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        [Browsable(
false)]
论Web控件开发 - 树状控件(一)        
public virtual bool IsTrackingViewState 
{
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void LoadViewState(object savedState) 
{
论Web控件开发 - 树状控件(一)            
if(savedState != null)
{    
论Web控件开发 - 树状控件(一)                Pair savedPair 
= (Pair)savedState;
论Web控件开发 - 树状控件(一)                ArrayList collectionState 
= (ArrayList)savedPair.First;
论Web控件开发 - 树状控件(一)                ArrayList collectionIndex 
= (ArrayList)savedPair.Second;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)                
for(int i = 0;i < collectionState.Count; i++)
{
论Web控件开发 - 树状控件(一)                    
int index = (int)collectionIndex[i];
论Web控件开发 - 树状控件(一)                    
if(index < Count)
论Web控件开发 - 树状控件(一)                        ((ViewStatePartBase)InnerList[index]).LoadViewState(collectionState[i]);
论Web控件开发 - 树状控件(一)                    
else
{
论Web控件开发 - 树状控件(一)                        ViewStatePartBase item 
= this.NewItem();
论Web控件开发 - 树状控件(一)                        item.LoadViewState(collectionState[i]);    
论Web控件开发 - 树状控件(一)                        AddItem(item);
论Web控件开发 - 树状控件(一)                    }

论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
protected abstract ViewStatePartBase NewItem();
论Web控件开发 - 树状控件(一)        
protected virtual int AddItem(ViewStatePartBase item)
{
论Web控件开发 - 树状控件(一)            
if(item!=null)
{
论Web控件开发 - 树状控件(一)                
if(this._isTrackingViewState)
{
论Web控件开发 - 树状控件(一)                    item.TrackViewState();
论Web控件开发 - 树状控件(一)                    item.SetDirty();
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                
this.InnerList.Add(item);
论Web控件开发 - 树状控件(一)                
return this.InnerList.Count - 1;
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
else
{
论Web控件开发 - 树状控件(一)                
throw new ArgumentNullException("item","item can't be null!");
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual object SaveViewState() 
{    
论Web控件开发 - 树状控件(一)            
if(this.Count > 0)
{
论Web控件开发 - 树状控件(一)                ArrayList collectionState;
论Web控件开发 - 树状控件(一)                ArrayList collectionIndex;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)                collectionState 
= new ArrayList();
论Web控件开发 - 树状控件(一)                collectionIndex 
= new ArrayList();
论Web控件开发 - 树状控件(一)                
for(int i=0;i< Count;i++)
{
论Web控件开发 - 树状控件(一)                    
object tmpState = ((ViewStatePartBase)InnerList[i]).SaveViewState();
论Web控件开发 - 树状控件(一)                    
if(tmpState!=null)
{
论Web控件开发 - 树状控件(一)                        collectionState.Add(tmpState);
论Web控件开发 - 树状控件(一)                        collectionIndex.Add(i);
论Web控件开发 - 树状控件(一)                    }
                    
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                
if(collectionState.Count > 0)
论Web控件开发 - 树状控件(一)                    
return new Pair(collectionState,collectionIndex);
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
return null;
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void TrackViewState() 
{
论Web控件开发 - 树状控件(一)            _isTrackingViewState 
= true;
论Web控件开发 - 树状控件(一)            
for(int i=0;i<Count;i++)
{
论Web控件开发 - 树状控件(一)                ((ViewStatePartBase)InnerList[i]).TrackViewState();
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        
public virtual bool IsEmpty
{

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
public virtual void SetDirty()
{
论Web控件开发 - 树状控件(一)            
for(int i = 0;i< Count ;i++)
论Web控件开发 - 树状控件(一)                ((ViewStatePartBase)InnerList[i]).SetDirty();
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion
论Web控件开发 - 树状控件(一)    }

论Web控件开发 - 树状控件(一)}

论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)


由代码可以看到,ViewStatePartCollectionBase从CollectionBase继承,这保证在设计器中会为此属性提供集合编辑器,而PersistenceMode(PersistenceMode.InnerProperty)属性则保证在页面序列化时会把其中的内容作为内部子属性来实现, 如下面Items属性所示:


而除了实现自身直接属性的视图状态管理以外还实现了其中包含的Items的状态管理,另外为了今后可以从用程序载入Item并支持视图状态管理,还实现了AddItem方法。为了简单,目前集合中只支持添加操作,并不支持插入、删除等操作,因为如果支持这些操作则还要考虑保存所有的Items中的视图状态情况比较烦锁,而一般来讲通常这些组件都是在第一次请求时装载,在回传情况下通过视图状态恢复,所以一般情况下已经够用。

下面是WinPopItem的实现

论Web控件开发 - 树状控件(一)using System;
论Web控件开发 - 树状控件(一)
using System.Web.UI;
论Web控件开发 - 树状控件(一)
using System.Web.UI.WebControls;
论Web控件开发 - 树状控件(一)
using System.Web.UI.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
using System.ComponentModel;
论Web控件开发 - 树状控件(一)
using System.ComponentModel.Design;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
namespace Keyss.WebControls
{    
论Web控件开发 - 树状控件(一)    [TypeConverter(
typeof(ExpandableObjectConverter))]
论Web控件开发 - 树状控件(一)    
public class WinPopItem:ViewStatePartBase
{

论Web控件开发 - 树状控件(一)        
public const string TargetUrlKey = "B";
论Web控件开发 - 树状控件(一)        
public const string TargetFrameKey = "C";
论Web控件开发 - 树状控件(一)        
public const string TargetFeaturesKey = "D";
论Web控件开发 - 树状控件(一)        
public const string ReplaceKey = "E";
论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        NotifyParentProperty(
true), 
论Web控件开发 - 树状控件(一)        Bindable(
true),
论Web控件开发 - 树状控件(一)        Category(
"Appearance"),
论Web控件开发 - 树状控件(一)        DefaultValue(
"")
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
public string TargetUrl 

论Web控件开发 - 树状控件(一)            
get 

论Web控件开发 - 树状控件(一)                
object o = ViewState[TargetUrlKey];
论Web控件开发 - 树状控件(一)                
return (o == null)?string.Empty:(string)o;
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)            
set 

论Web控件开发 - 树状控件(一)                ViewState[TargetUrlKey] 
= value; 
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        NotifyParentProperty(
true), 
论Web控件开发 - 树状控件(一)        Bindable(
true),
论Web控件开发 - 树状控件(一)        Category(
"Appearance"),
论Web控件开发 - 树状控件(一)        DefaultValue(
"")
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
public string TargetFrame 

论Web控件开发 - 树状控件(一)            
get 

论Web控件开发 - 树状控件(一)                
object o = ViewState[TargetFrameKey];
论Web控件开发 - 树状控件(一)                
return (o == null)?string.Empty:(string)o;
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)            
set 

论Web控件开发 - 树状控件(一)                ViewState[TargetFrameKey] 
= value; 
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        NotifyParentProperty(
true), 
论Web控件开发 - 树状控件(一)        Bindable(
true),
论Web控件开发 - 树状控件(一)        Category(
"Appearance"),
论Web控件开发 - 树状控件(一)        DefaultValue(
"")
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
public string TargetFeatures 

论Web控件开发 - 树状控件(一)            
get 

论Web控件开发 - 树状控件(一)                
object o = ViewState[TargetFeaturesKey];
论Web控件开发 - 树状控件(一)                
return (o == null)?string.Empty:(string)o;
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)            
set 

论Web控件开发 - 树状控件(一)                ViewState[TargetFeaturesKey] 
= value; 
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        DefaultValue(
true),
论Web控件开发 - 树状控件(一)        NotifyParentProperty(
true), 
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
public bool Replace 

论Web控件开发 - 树状控件(一)            
get 

论Web控件开发 - 树状控件(一)                
object o = ViewState[ReplaceKey];
论Web控件开发 - 树状控件(一)                
return (o == null)?true:(bool)o;
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)            
set 

论Web控件开发 - 树状控件(一)                ViewState[ReplaceKey] 
= value; 
论Web控件开发 - 树状控件(一)            }
 
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void CopyFrom(WinPopItem s)
{    
论Web控件开发 - 树状控件(一)            
base.CopyFrom(s);
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual void MergeWith(WinPopItem s)
{
论Web控件开发 - 树状控件(一)            
base.MergeWith(s);
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        [EditorBrowsable(EditorBrowsableState.Never)]
论Web控件开发 - 树状控件(一)        
public virtual string GetPreRenderJScript()
{
论Web控件开发 - 树状控件(一)
//            window.open(url, target, features, replace);
论Web控件开发 - 树状控件(一)
            return string.Format("window.open({0},{1},{2},{3})",new object[]
{
论Web控件开发 - 树状控件(一)                    RenderHelper.StrToJavaPara(
this.TargetUrl),
论Web控件开发 - 树状控件(一)                    RenderHelper.StrToJavaPara(
this.TargetFrame),
论Web控件开发 - 树状控件(一)                    RenderHelper.StrToJavaPara(
this.TargetFeatures),
论Web控件开发 - 树状控件(一)                    RenderHelper.BoolToJavaPara(
this.Replace)
论Web控件开发 - 树状控件(一)                }
+ ";\n\r";
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        
public virtual void LoadFromDataRow(System.Data.DataRow row)
{
论Web控件开发 - 树状控件(一)            System.Data.DataTable table 
= row.Table;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)            
if(table.Columns.Contains("TargetURL"))
{
论Web控件开发 - 树状控件(一)                
this.TargetUrl = (string)row["TargetURL"];
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
if(table.Columns.Contains("TargetFrame"))
{
论Web控件开发 - 树状控件(一)                
this.TargetFrame = (string)row["TargetFrame"];
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
if(table.Columns.Contains("TargetFeatures"))
{
论Web控件开发 - 树状控件(一)                
this.TargetFeatures = (string)row["TargetFeatures"];
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
if(table.Columns.Contains("Replace"))
{
论Web控件开发 - 树状控件(一)                
this.Replace = (bool)row["Replace"];
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion
论Web控件开发 - 树状控件(一)    }

论Web控件开发 - 树状控件(一)}

论Web控件开发 - 树状控件(一)
WinPopItem的实现比较简单,只是在基类上增加了一些自定义属性,并且增加一个GetPreRenderJScript() 函数用来用属性生成前台的jscript程序字符串。而LoadFromDataRow则用来从一个tablerow中加载属性。

以下是WinPopItemCollection的源代码
论Web控件开发 - 树状控件(一)using System;
论Web控件开发 - 树状控件(一)
using System.Web.UI;
论Web控件开发 - 树状控件(一)
using System.Web.UI.WebControls;
论Web控件开发 - 树状控件(一)
using System.Web.UI.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing.Design;
论Web控件开发 - 树状控件(一)
using System.Drawing;
论Web控件开发 - 树状控件(一)
using System.Collections;
论Web控件开发 - 树状控件(一)
using System.ComponentModel;
论Web控件开发 - 树状控件(一)
using System.ComponentModel.Design;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
namespace Keyss.WebControls
{
论Web控件开发 - 树状控件(一)    [ 
论Web控件开发 - 树状控件(一)    PersistenceMode(PersistenceMode.InnerProperty),
论Web控件开发 - 树状控件(一)    ] 
论Web控件开发 - 树状控件(一)    
public class WinPopItemCollection:ViewStatePartCollectionBase
{
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        
public int Add(WinPopItem item)
{
论Web控件开发 - 树状控件(一)            
return base.AddItem(item);
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
public WinPopItem this[int index]
{
论Web控件开发 - 树状控件(一)            
get 
{
论Web控件开发 - 树状控件(一)                
if(index >= Count || index < 0)
{
论Web控件开发 - 树状控件(一)                    
return null;
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                
return (WinPopItem)this.InnerList[index];
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
protected override ViewStatePartBase NewItem()
{
论Web控件开发 - 树状控件(一)            
return new WinPopItem();
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)    }

论Web控件开发 - 树状控件(一)}

论Web控件开发 - 树状控件(一)
WinPopItemCollection的实现了集合的Add方法,当页面解析时将调用此方法从设计时的Tag标记装载Items ,并且定义了一个索引器,另外还重载了基类中的NewItem用来返回WinPopItem类型。

最后是WinPop组件的实现
论Web控件开发 - 树状控件(一)using System;
论Web控件开发 - 树状控件(一)
using System.Web.UI;
论Web控件开发 - 树状控件(一)
using System.Web.UI.WebControls;
论Web控件开发 - 树状控件(一)
using System.ComponentModel;
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)
namespace Keyss.WebControls
{
论Web控件开发 - 树状控件(一)    [DefaultProperty(
"Items"), 
论Web控件开发 - 树状控件(一)        ToolboxData(
"<{0}:WinPop runat=server></{0}:WinPop>")]
论Web控件开发 - 树状控件(一)    
public class WinPop : System.Web.UI.WebControls.WebControl
{

论Web控件开发 - 树状控件(一)        
public virtual void LoadFromTable(System.Data.DataTable table)
{
论Web控件开发 - 树状控件(一)            
try
{
论Web控件开发 - 树状控件(一)                
this.Items.Clear();
论Web控件开发 - 树状控件(一)                
foreach(System.Data.DataRow row in table.Rows)
{
论Web控件开发 - 树状控件(一)                    Keyss.WebControls.WinPopItem item 
= new Keyss.WebControls.WinPopItem();
论Web控件开发 - 树状控件(一)                    item.LoadFromDataRow(row);
论Web控件开发 - 树状控件(一)                    
this.Items.Add(item);
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
catch
{
论Web控件开发 - 树状控件(一)                
throw new ArgumentException("Table or filed name error!",table.TableName);
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        
private WinPopItemCollection _items;
论Web控件开发 - 树状控件(一)        [
论Web控件开发 - 树状控件(一)        NotifyParentProperty(
true),
论Web控件开发 - 树状控件(一)        PersistenceMode(PersistenceMode.InnerProperty),
论Web控件开发 - 树状控件(一)        DesignerSerializationVisibility(DesignerSerializationVisibility.Content),
论Web控件开发 - 树状控件(一)        ]
论Web控件开发 - 树状控件(一)        
public WinPopItemCollection Items
{
论Web控件开发 - 树状控件(一)            
get 
{
论Web控件开发 - 树状控件(一)                
if(this._items ==null
{
论Web控件开发 - 树状控件(一)                    _items 
= new WinPopItemCollection();
论Web控件开发 - 树状控件(一)                    
if(this.IsTrackingViewState)
论Web控件开发 - 树状控件(一)                        _items.TrackViewState();
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                
return _items;
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        
protected override void OnPreRender(EventArgs e)
{    
论Web控件开发 - 树状控件(一)            System.Text.StringBuilder codesource 
= new System.Text.StringBuilder();
论Web控件开发 - 树状控件(一)            
if((this.Enabled)&&(!this.Items.IsEmpty)&&(!Page.IsStartupScriptRegistered(this.ClientID)))
{
论Web控件开发 - 树状控件(一)                codesource.Append(
"<script language=\"JavaScript\" >");
论Web控件开发 - 树状控件(一)                codesource.Append(
"\n\r");
论Web控件开发 - 树状控件(一)                
for(int i=0;i<Items.Count;i++)
{
论Web控件开发 - 树状控件(一)                    codesource.Append(Items[i].GetPreRenderJScript());
论Web控件开发 - 树状控件(一)                }

论Web控件开发 - 树状控件(一)                codesource.Append(
"</script>");
论Web控件开发 - 树状控件(一)                Page.RegisterStartupScript(ClientID,codesource.ToString());
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
protected override void Render(HtmlTextWriter writer)
{
论Web控件开发 - 树状控件(一)            
if((this.Site!=null)&&this.Site.DesignMode)
论Web控件开发 - 树状控件(一)                writer.WriteLine(
this.ID);
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion

论Web控件开发 - 树状控件(一)        
protected override void LoadViewState(object state)
{
论Web控件开发 - 树状控件(一)            
if(state !=null)
{
论Web控件开发 - 树状控件(一)                
object[] savedState = (object[])state;
论Web控件开发 - 树状控件(一)                
if(savedState[0!= null)
论Web控件开发 - 树状控件(一)                    
base.LoadViewState(savedState[0]);
论Web控件开发 - 树状控件(一)                
if(savedState[1!= null)
论Web控件开发 - 树状控件(一)                    Items.LoadViewState(savedState[
1]);
论Web控件开发 - 树状控件(一)            }
    
论Web控件开发 - 树状控件(一)
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
protected override object SaveViewState()
{
论Web控件开发 - 树状控件(一)            
object[] savedState = new object[2];
论Web控件开发 - 树状控件(一)            savedState[
0= base.SaveViewState ();
论Web控件开发 - 树状控件(一)            savedState[
1= Items.SaveViewState();
论Web控件开发 - 树状控件(一)            
for(int i=0;i<savedState.Length;i++)
{
论Web控件开发 - 树状控件(一)                
if(savedState[i] != null)
论Web控件开发 - 树状控件(一)                    
return savedState;
论Web控件开发 - 树状控件(一)            }

论Web控件开发 - 树状控件(一)            
return null;
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
protected override void TrackViewState()
{
论Web控件开发 - 树状控件(一)            
base.TrackViewState();
论Web控件开发 - 树状控件(一)            Items.TrackViewState();
论Web控件开发 - 树状控件(一)        }

论Web控件开发 - 树状控件(一)        
#endregion
论Web控件开发 - 树状控件(一)    }

论Web控件开发 - 树状控件(一)}

论Web控件开发 - 树状控件(一)

从源程序中可以看到WinPop中含有一个默认属性Items并且重载了基类中的视图状态函数,用来支持Items的视图状态。为了加载方便还实现了LoadFromTable方法用来从datatable中载入Items数据。由于WinPop并没有实际的HTML元素,只是在页面中注册jscript角本,所以重载的Render方法只是判断是否为设计时,如果是则输出控件的ID便于设计时选择该控件,而在OnPreRender方法中则注册startup类型的jscript块,这样当页面绘制时,控件会根据Items中的内容绘制一系列的window.open(url, target, features, replace);来实现弹出窗口的目的。

相关文章:

  • 2021-09-28
  • 2021-08-04
  • 2021-07-16
  • 2021-12-30
  • 2021-07-04
  • 2021-10-29
猜你喜欢
  • 2021-10-29
  • 2022-12-23
  • 2021-09-28
  • 2021-07-11
  • 2021-05-31
  • 2021-10-19
  • 2021-11-06
相关资源
相似解决方案