zoukankan      html  css  js  c++  java
  • 探讨微软ASP.NET AJAX控件开发技术(客户端)

    本系列文章将通过具体的实例从客户端和服务端两个角度全面探讨ASP.NET AJAX框架中的控件(Control)开发所涉及的技术。

      一、 简介

      如今,ASP.NET AJAX框架以其与ASP.NET 2.0系统的有机整合与完全面向对象的客户端JavaScript组件模型正在吸引着越来越多的Web开发人员。此外,这个框架还为基于ASP.NET 2.0平台的AJAX Web开发提供了一揽子方案(尽管尚嫌稚气)。

      首先,我们来回忆ASP.NET AJAX框架设计的主要目标:其一,扩展现有的ASP.NET服务器端模型,让其能够生成支持富客户端的JavaScript代码;其二,为ASP.NET应用增加客户端编程模型,让纯粹的客户端编程变得更为容易。

      基于此,我们至少可以从如下两个方面来扩展ASP.NET AJAX框架。第一,创建更为丰富的ASP.NET AJAX服务器端扩展器控件,以便为这些控件提供丰富的客户端Ajax行为。此部分的极好例证就是AJAX Control Toolkit中的大部分控件。第二,创建纯粹的客户端组件/控件。这部分的例证就是框架已经实现的对于常规HTML元素的封装、高级客户端控件ListView及客户端离线数据源DataSource等。为此,控件开发者可以创建新的客户端组件/控件来封装典型的客户端功能,或者基于现有客户端组件/控件加以扩展。

      【说明】第一,事实上,我们可以进一步沿着ASP.NET AJAX客户端与服务器端架构层次关系图进一步扩展其底层。第二,要开发ASP.NET AJAX控件要求开发人员首先具备关于ASP.NET AJAX框架的客户端和服务器器知识(相比而言,在具体编程的过程中要求掌握更多的客户端相关知识)。

      在本文中,我们将从客户端和服务端两个角度全面探讨ASP.NET AJAX框架中的控件开发相关技术。首先,让我们从客户端开始。

      二、 控件开发客户端相关技术

    微软Ajax库的最伟大之处在于,它扩展了JavaScript的面向对象编程模型并且提供一个增强的类型系统(其中包含命名空间,类,接口,枚举,异常处理,反射及其它为.NET开发者所熟悉的结构)。

      尽管Ajax控件都是继承自Sys.UI.Control,但是,整个Ajax控件的核心却是Sys.Comoponent类。这个类模拟了.NET框架的System.ComponentModel.Comoponent类。图1展示了在Ajax控件开发中继承类之间关系的示意图。

      探讨微软ASP.NET AJAX控件开发技术(客户端)

      图1.Ajax控件开发中所涉及的类关系图。

      (一) Sys.IDisposable

      这个接口类似于.NET框架的IDisposable接口。组件利用的所有资源都应该在dispose方法中释放。具体地说,这些资源包括事件处理器,大型数组,等等。

      (二) Sys.INotifyDisposing

      这个接口允许组件用户通过disposing事件检测该组件的释放情况。

      (三) Sys.INotifyPropertyChange

      这个接口允许组件用户通过PropertyChanged事件检测有关属性的改变情况。在本文后面,我们还要对此展开详细的讨论。

      (四) Sys.Component

      该Sys.Component类要实现上面所有的接口。封装了复杂的逻辑或包含子DOM元素的组件一般都要求使用一个中央位置用于初始化和清除类的实例。这通常是通过重载Sys.Component类的initialize和dispose方法实现的。下列图2展示了Sys.Component类的主要功能。

      探讨微软ASP.NET AJAX控件开发技术(客户端)

    图2—Sys.Component类主要功能展示。

      进一步归纳来看,共存在两种类型的组件:

       Nonvisual(非可视化的)

       Visual(可视化的)

      一个非可视化的组件没有任何用户接口。一个典型的例子是ASP.NETAJAX框架内置的定时器控件或一个定制的对web服务进行队列调用的组件。非可视化组件类似于ASP.NET中的ObjectDataSource和TableAdapter控件,但是没有任何用户接口。

      另一方面,可视化的组件则提供了一个用户接口。一个典型的例子是Ajax Control Toolkit中的控件,UpdateProgress,以及AjaxGrid控件等等。其实,我们还可以把可视化组件进一步划分为:

       Action(Sys.UI.Action)

       Control(Sys.UI.Control)

      (五) Sys.UI.Action

      一个行为(Action)的目的是用来扩展一个DOM元素而不改变它的核心功能。Ajax Control Toolkit中的大多数控件就属于行为,例如该AutoCompleteTextBox,MaskEdit,DragPanel等。一个行为必须拥有一个相关联的DOM元素。注意,单个DOM元素可以拥有多个与之相关联的行为。既然本文的重点在讨论控件的开发技术,所以,我们不再进一步讨论行为的概念。如果你对此感兴趣的话,请访问Ajax Control Toolkit站点。

      (六) Sys.UI.Control

      相对于上面的Action而言,一个Control(控件)本身就是一个DOM元素。创建一个控件的主要目的是通过对一个现有控件加以包装进而提供新的功能。其典型的示例就是UpdatePanel,UpdateProgress或Ajax Control Toolkit中新加入的Tab控件。下列的代码向你展示了创建一个控件所要求的最少代码。

      列表1.构建一个客户端控件至少需要实现的JavaScript框架代码。

    Type.registerNamespace('DummyNamespace');
    DummyNamespace.DummyControl = function(element)
    {
    DummyNamespace.DummyControl.initializeBase(this, [element]);
    }
    DummyNamespace.DummyControl.prototype =
    {
    initialize : function()
    {
    DummyNamespace.DummyControl.callBaseMethod(this, 'initialize');
    },
    dispose : function()
    {
    DummyNamespace.DummyControl.callBaseMethod(this, 'dispose');
    }
    }
    DummyNamespace.DummyControl.registerClass('DummyNamespace.DummyControl',
    Sys.UI.Control);
    if (typeof (Sys) != 'undefined')
    {
    Sys.Application.notifyScriptLoaded();
    }

    上面所列出的仅是一个控件的框架代码。如你所见,与之相关联的DOM元素作为参数传入该控件的构造器函数中。我们还要重载initialize和dispose方法以便执行初始化和最后释放控件时的清理工作。因为在此列出的仅是一个虚构的控件,所以,除了调用基类的一些方法外我们什么也没有做。下列图3展示了Sys.UI.Control类提供的主要功能。

      探讨微软ASP.NET AJAX控件开发技术(客户端)

       图3—Sys.UI.Control类提供的方法。

      一个控件和一个行为之间的另一个区别在于,一个行为允许你设置一个id而一个控件却不允许。事实上,一个控件的id与其相关联的DOM元素是一致的。下面是这个类中常用方法(有些是父类提供)的详细解释。

     get_element():返回该控件描述的DOM元素。

       get_id():返回该控件的id,为$find语句所用以引用此控件。

       set_id():如果你试图设置id,那么,你将得到一个Error.invalidOperation异常,因为控件是不允许设置id的。

       get_parent():返回父级控件。

       set_parent():设置父控件。

       get_visibilityMode():visibilityMode是一个枚举,其取值是hide或collapse。

       set_visibilityMode():设置visibilityMode的值。

       get_visibile():返回相应DOM元素的可见性。

       set_visibile():设置DOM元素风格可见性,取值为hidden或visible。

       addCssClass():把指定的CSS类添加到DOM元素的className中。

       dispose():继承自Sys.Component。

       initialize():继承自Sys.Component。

       onBubbleEvent():处理由raiseBubbleEvent激发的事件。如果这个方法返回true,则该事件将不会被上传到其父元素中进行处理。注意,如果你不处理该事件的话,你应该把此事件交由父级来作默认处理。

       raiseBubbleEvent():引发一个事件并交由父控件处理。总的来看,当你创建复杂的控件(经常是包含一个或多个子控件并且想把一个事件从子控件上交由其父控件来处理时)时往往要使用onBubbleEvent与这个方法。

       removeCssClass():从DOM元素的className中删除指定的CSS类。

       toggleCssClass():把指定的CSS类添加到DOM元素的className中(如果以前没有设置的话)。注意,如果已经指定了这个CSS类,那么,将从className中移除该类。

    三、 实例分析

      在本示例中,我们将创建一个增强的客户端ImageButton控件。这个控件在鼠标移过时将显示一幅不同的图像。

      (一)创建示例AJAX网站

      启动Visual Studio 2005,选择“文件→新建网站…”,然后选择“ASP.NET AJAX-Enabled Web Site”模板,命名工程为“AjaxClientCtrlTest”,并选择C#作为内置支持语言,最后点击OK。

      注意,此后系统将自动加入对于程序集System.Web.Extensions.dll的引用(因其被自动加入到GAC中,所以默认情况下无法直接看到)。当然,你还会注意到,作为ASP.NET AJAX控制中心的服务器控件ScriptManager被自动地添加到默认网页Default.aspx中。然后,把一个HTML <image>元素拖动到此页面中,并稍加修改,得到如下图4所示布局。

      探讨微软ASP.NET AJAX控件开发技术(客户端)

      图4.示例网站屏幕快照。

      (二)使用面向对象JavaScript设计控件类

      以鼠标右击工程添加一个JavaScript脚本文件ImageButton.js,内容如下列表2所示。

      列表2—脚本控件MyCliImageButton完整源码。  

    Type.registerNamespace('AjaxImageButtonNamespace');
    AjaxImageButtonNamespace.MyCliImageButton = function(element)
    {
    this._hoverImageUrl = '';
    this._originalImageUrl = '';
    this._mouseOverHandler = null;
    this._mouseOutHandler = null;
    this._clickHandler = null;
    AjaxImageButtonNamespace.MyCliImageButton.initializeBase(this, [element]);
    }
    AjaxImageButtonNamespace.MyCliImageButton.prototype =
    {
    get_hoverImageUrl : function(){
    return this._hoverImageUrl;
    },
    set_hoverImageUrl : function(value) {
    var e = Function._validateParams(arguments, [{name: 'value', type: String}]);
    if (e) throw e;
    if (this._hoverImageUrl != value)
    {
    this._hoverImageUrl = value;
    this.raisePropertyChanged('hoverImageUrl');
    }
    },
    initialize : function(){
    AjaxImageButtonNamespace.MyCliImageButton.callBaseMethod(this, 'initialize');
    var target = this.get_element();
    this._originalImageUrl = target.src;
    this._mouseOverHandler = Function.createDelegate(this, this._onMouseOver);
    this._mouseOutHandler = Function.createDelegate(this, this._onMouseOut);
    this._clickHandler = Function.createDelegate(this, this._onClick)
    $addHandlers(target, {'mouseover':this._mouseOverHandler, 'mouseout':this._mouseOutHandler, 'click': this._clickHandler}, this);
    },
    dispose : function(){
    $clearHandlers(this.get_element());
    delete this._mouseOverHandler;
    delete this._mouseOutHandler;
    delete this._clickHandler;
    AjaxImageButtonNamespace.MyCliImageButton.callBaseMethod(this, 'dispose');
    },
    add_click : function(handler) {
    this.get_events().addHandler('click', handler);
    },
    remove_click : function(handler) {
    this.get_events().removeHandler('click', handler);
    },
    _onMouseOver : function(e) {
    e.target.src = this._hoverImageUrl;
    },
    _onMouseOut : function(e) {
    e.target.src = this._originalImageUrl;
    },
    _onClick : function(e) {
    e.preventDefault();
    var handler = this.get_events().getHandler('click');
    if (handler != null) {
    handler(this, Sys.EventArgs.Empty);
    }
    }
    }
    AjaxImageButtonNamespace.MyCliImageButton.registerClass('AjaxImageButtonNamespace.MyCliImageButton', Sys.UI.Control);
    if (typeof(Sys) != 'undefined')
    Sys.Application.notifyScriptLoaded();
    }

      如你所见,我们重载了initialize方法来“钩住”对应DOM元素的mouseover,mouseout和click事件,以便我们可以在这些事件中设置合适的图像并激发click事件。我们还重载了dispose方法以分离我们在initialize方法中设置的事件处理器。后面,我们还要细致讨论。

    (三)使用控件

      有了上面的控件类,至于使用就很简单了。请切换到页面default.aspx的源码视图,然后在<head>区创建如下列表3所示内容。

      列表3 

    <script type="text/javascript">
    function pageLoad(){
    $create(AjaxImageButtonNamespace.MyCliImageButton, {'hoverImageUrl':'Images/updateh.gif'}, {'click':buttonClicked}, null, $get('cliBtn'));
    }
    function buttonClicked(sender, e) {
    alert('I am clicked');
    return false;
    }
    </script>

      在此,我们使用ASP.NET AJAX客户端全局方法$create创建控件AjaxImageButtonNamespace.MyCliImageButton的一个实例,指定其属性hoverImageUrl取值、click事件的处理器函数,并把它与当前页面中的HTML <IMAGE>元素关联起来。当然,为了使得框架在运行时能够找到我们刚才建立的脚本文件,还要对ScriptManager的属性稍微设置一下,如下列表4所示。

      列表4—在ScriptManager控件内配置脚本文件。

      <asp:ScriptManager ID="ScriptManager1" runat="server" >

      <Scripts>

      <asp:ScriptReference Path="ImageButton.js" />

      </Scripts>

      </asp:ScriptManager>

      现在,请按F5键运行此页面观察结果即可。下图5相应于当鼠标移动到图像按钮上时发生的变化(你还可以点击之自行观察效果)。

      探讨微软ASP.NET AJAX控件开发技术(客户端)

    图5.当鼠标移动到图像按钮上时图像切换成另一幅。

      下面,我们针对使用ASP.NET AJAX框架提供的面向对象JavaScript进行客户端控件编程时有关概念作全面分析。

      四、 使用JavaScript进行客户端控件编程相关概念

      (七) 属性

      把属性添加到一个控件中是很简单的。在此,推荐使用的方法是在控件构造器中声明一个私有变量,并在prototype部分添加相应的getter及setter属性方法。当访问该属性时,ASP.NET AJAX框架总是添加get_和set_前缀。因此,当添加getter/setter时,我们应该遵循get_propertyName和set_propertyName命名惯例。正如在上面的示例中相应于hoverImageUrl属性所实现的,我们使用get_hoverImage()作为getter,set_hoverImage()作为setter。但是在上面的例子中当我们使用一个$create()语句创建该控件时,我们仅传递了hoverImageUrl,因为ASP.NET AJAX框架将会自动添加set_前缀。与属性有关的另外一个重要问题是,当它的属性改变时如何通知该控件的客户端。下列的代码片断展示了如何激发propertyChanged事件。

    set_hoverImageUrl : function(value){
    if (this._hoverImageUrl != value)
    {
    //如果不同于当前值则仅设置新值
    this._hoverImageUrl = value;
    //激发propertyChanged事件
    //注意这是一个位于基类Sys.Component中的方法
    this.raisePropertyChanged('hoverImageUrl');
    }
    }

      (八) 方法

      使用JavaScript编写一个类的方法时,一般不作特殊考虑;但是,我们还是推荐在控件的原型(prototype)对象中添加它们。

      【注】就JavaScript本身编程而言,其表达形式多样,但建议你遵循这里推荐的格式(也正是框架相应js源码中所使用的格式)。

    (九) 事件

      在控件类中添加事件类似添加属性。首先,在控件构造器中把事件处理器声明为一个私有变量并在原型对象中创建相应于该事件的一个add/delete方法对。尽管并不要求在模块级声明事件处理器变量,但是,这样做的确存在一些优点,我将在Ajax框架的Function.createDelegate()节讨论这个问题。当添加/删除事件时,Ajax框架总是自动地添加上add_和remove_前缀。因此,当添加事件订阅代码时,该方法应该以add_eventName方式命名;反之,以remove_eventName方式命名。例如,在上面的click事件示例中,我们以add_click和remove_click形式添加事件订阅方法,但是在$create语句中,我们以click方式对它命名,由Ajax框架负责添加add_前缀。如你在上面的示例所见,add_click和remove_click都引用了this.get_event()(继承自Sys.Component)。这个Sys.Component类维持着一个订阅事件的内部列表,存储于一个Sys.EventHandlerList类的实例(以及各自它们的处理器)中。

      除了这个get_event()方法对外,还推荐使用raiseEventName方法来激发事件。由于上面所举的仅是一个很基本的例子,所以,我没有在其中使用它。但是,当你想从多处激发同一个事件时你必须添加它。当激发一个事件时,我们可以传递一些称为事件参数的上下文数据。

      在事件中传递参数的推荐的方法是,创建一个继承自Sys.EventArgs的新类并构造器中传递要求的参数。针对每一个参数,它将在原型对象部分暴露getter和setter。因为本例非常简单,所以,没有使用这种更一般的方式。

      当前在该微软Ajax库中相应于事件参数存在两个内置类—Sys.EventArgs和Sys.CancelEventArgs。即使你有一个不需要任何事件参数的事件,那么,你也应该传递Sys.EventArgs.Empty(例如在上面的ImageButton控件示例中),以便事件订阅者可以遵循相同的模式来处理所有的事件。

    如果你细致研究框架源码,你会注意到,其中绝大部分的控件都遵循了相同的模式—使用函数HandleEvent(sender, e)来处理所有事件。注意,这与.NET框架处理事件所实现的模式是相同的。而且,遵循这种方式给人的初始感觉过于复杂,但是却更遵循了一般性规律,便于以后的更复杂设计。

      五、 其它几个重要的全局客户端方法   下列,我们列出在开发客户端控件时,一些常用(有些是必用的)的重要方法。

      $get()

      这个方法是我们以前常用的Document.getElementById方法的一个快捷方式。此方法使用一个可选的父级DOM元素;如果没有指定这个父元素,那么,搜索操作将从Document对象开始执行。

      $create()

      这个方法是Sys.Component.Create方法的一个快捷方式,用于实例化一个客户端组件,行为或控件。这个方法的完整定义如下所示:

      $create(type, properties, events, references, element);

      举例:

      $create(AjaxImageButtonNamespace.MyCliImageButton, {'hoverImageUrl':'Images/updateh.gif'}, {'click':buttonClicked}, null, $get('cliBtn'));

      在此,第一个参数使用了对象的完整类型名。例如,当创建一个MyCliImageButton控件时,我们需要传递AjaxImageButtonNamespace.MyCliImageButton。这个参数必须存在。

      第二个参数对应一个键/值对(每一个一个键/值对对应一个特定属性及其该属性的取值)数组。例如,当创建MyCliImageButton控件时,我们需要要传递hoverImageUrl并给它赋值为{'hoverImageUrl':' Images/update-h.gif'}。如果我们想设置更多的属性,我们需要使用一个逗号来隔开这些键/值对。例如

  • 相关阅读:
    shell脚本中判断上一个命令是否执行成功
    nginx 414 Request-URI Too Large
    nginx 重写URL尾部斜杠
    Linux shell 日期,时间相关的命令
    shell脚本中自定义日志记录到文件
    scanf后面跟一个getchar
    1.Tarball软件make与makefile详解(还需要补充)
    <>和“”的区别
    malloc,calloc,realloc,alloc
    toString()方法细节
  • 原文地址:https://www.cnblogs.com/dobit/p/1746248.html
Copyright © 2011-2022 走看看