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

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

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


以下是ViewStatePartBase的源代码

using System;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.Design;
using System.Drawing.Design;
using System.Drawing;
using System.ComponentModel;
using System.ComponentModel.Design;
using System.Collections;

namespace Keyss.WebControls
{
    
public abstract class ViewStatePartBase:IStateManager
    
{
        
viewstate
        
helper
    }

}


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

以下是ViewStatePartBaseCollection的源代码

using System;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.Design;
using System.Drawing.Design;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.ComponentModel.Design;

namespace Keyss.WebControls
{
    [ 
    PersistenceMode(PersistenceMode.InnerProperty),
    ] 
    
public abstract class ViewStatePartCollectionBase:System.Collections.CollectionBase,IStateManager
    
{
        
viewstate
        
helper function
    }

}



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


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

下面是WinPopItem的实现

using System;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.Design;
using System.Drawing.Design;
using System.Drawing;

using System.ComponentModel;
using System.ComponentModel.Design;

namespace Keyss.WebControls
{    
    [TypeConverter(
typeof(ExpandableObjectConverter))]
    
public class WinPopItem:ViewStatePartBase
    
{
        
const
        
field
        
copy merge
        
prerender
        
load from table
    }

}

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

以下是WinPopItemCollection的源代码
using System;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.Design;
using System.Drawing.Design;
using System.Drawing;
using System.Collections;
using System.ComponentModel;
using System.ComponentModel.Design;

namespace Keyss.WebControls
{
    [ 
    PersistenceMode(PersistenceMode.InnerProperty),
    ] 
    
public class WinPopItemCollection:ViewStatePartCollectionBase
    
{

        
public int Add(WinPopItem item)
        
{
            
return base.AddItem(item);
        }

        
public WinPopItem this[int index]
        
{
            
get 
            
{
                
if(index >= Count || index < 0)
                
{
                    
return null;
                }

                
return (WinPopItem)this.InnerList[index];
            }

        }

        
protected override ViewStatePartBase NewItem()
        
{
            
return new WinPopItem();
        }


    }

}

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

最后是WinPop组件的实现
using System;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.ComponentModel;

namespace Keyss.WebControls
{
    [DefaultProperty(
"Items"), 
        ToolboxData(
"<{0}:WinPop runat=server></{0}:WinPop>")]
    
public class WinPop : System.Web.UI.WebControls.WebControl
    
{
        
loaddata from table
        
items
        
render
        
view State
    }

}


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