5.3  集成资源文件
虽然在上一节中,我们控制了相同的脚本只在相同页面呈现一遍,但将大段相同的脚本重复呈现到 不同的页面也是一种网络与服务器资源的浪费,因为这种方式完全没有利用HTTP的缓存机制,下面我们将探讨如何克服这种问题。
5.3.1  外部JS文件与部署
利用缓存最简单的办法就是把相同的代码独立到一个文件中,比如AutoFlex.js,IIS能自动处理静态文件的Http-Cache策略,它会要求客户端把请求的静态文件缓存起来,那么客户端下次请求相同文件时,就可以利用缓存而不需从服务器上把文件下载下来。
所以,您可以把上一节中实现的AutoFlex客户端组件的代码分离到专门的一个文件中,然后用ClientScriptManager类的 RegisterClientScriptInclude()方法将外部脚本文件注册到页面中。
*************************************
但另一个问题又出现了,我们怎么把这个脚本文件放到用户的网站项目中去?
传统的做法是新建一个Web安装项目,将脚本文件生成到用户的网站根目录下,如C:\inetpub\wwwroot\aspnet_client,ASP.NET1.0+就是用这种方式部署验证控件所需的脚本
**************************************
但这种方式非常脆弱,只要我们需要更改网站根目录的位置,或在IIS6.0+中部署多个站点,这种方式都无能为力,所以ASP.NET2.0引入了一种新的方式,即将脚本编译成装配件中的集成资源,然后用axd后缀路径输出资源文件。
****************************************
接下来我们再次增强TextArea的功能,让它支持输入Tab键。在默认情况下,Tab键会让焦点跳到下一个可控制控件,比如按钮、链接。
不过这一次我们不从零开始编写脚本,而是借助于强大的JQuery脚本库和它最棒的插件Interface。
**************************************
5.3.2  JQuery简介
AJAX火起来后,编写JavaScript的需求迅速膨胀,于是互联网上涌现出大量优秀的JS脚本库,比如prototype.js,它的Ruby风格就曾深深地吸引了我,但不久,我就“移情别恋”了,因为我发现JQuery更让人感到亲切。一直到现在,我都把它当成最顺手的JS工具,就算使用ASP.NET AJAX框架做组件框架时,也会大量使用JQuery来实现功能。
与其他JavaScript框架(也许有些人更喜欢说成AJAX框架)相比,JQuery有如下特点:
*************************************************
(1)对DOM结构更多样的操作,JQuery允许你根据DOM元素的ID,Class,标签名等查找页面元素,更重要的是我们还可以用CSS(甚至是CSS3)和XPath语法来组合ID,Class等完成更复杂的对象选择。
比如要在异步更新页面时禁用页面中的所有输入框——$(“input[@type= text]”).attr(‘disabled’,’disabled’),
隐藏所有可见的DIV——$(“div:visible”).hide(),
选中的单选按钮——$(“input[@type=radio][@checked]”),
再比如P下面的A——$(‘p/a’)。
*******************************************
(2)语句链。JQuery对象的方法在执行后仍然返回JQuery对象(只有少数方法会改变JQuery对象所包含的元素),比如,
$(“#boodsTable tr”).mouseover(trMouseOverHandler).mouseout (trMouseOutHandler),
可以用链式的语句为boodsTable内的TR元素添加MouseOver事件处理函数和MouseOut事件处理函数。 (3)模拟还未在浏览器中实现的CSS3功能。 (4)优秀的AJAX支持和跨浏览器支持。 http://jquery.com JQuery还有一大批相当优秀的插件,interface是其中最优秀的之一。 Interface不仅包含一组不错的“效果”,还有很多功能强大的组件,更多详情请访问 http://interface.eyecon.ro/。 ********************************************
5.3.3  TabbableTextArea
很多次遭遇在想输入一个制表符时,Tab键却把焦点转到了下一个控件中的尴尬。现在好了,JQuery的Interface插件已经解决了这个问题,而且只需一行代码:$(element).EnableTabs()。
所以要完成TabbableTextArea控件并不需要写多少代码,关键在于我们怎么把jquery.js和interface.js这两个脚本文件集成进来。
******************************************
这一次,我们将采用内嵌资源的方式把脚本资源提供给 用 户(应该是“程序”)。
这种方式将控件所需脚本、图片等资源集成在控件所在的 装配件中,而不需要 用户(应该是“我们”)把控件依赖的文件安装到 某个特定的目录,控件使用者甚至感觉不到控件依赖于某些文件,所以说它是最方便 用户(应该是“我们”)的。
********************
要将资源内嵌到装配件中非常简单,将文件包含在项目中,在解决方案资源管理器窗口 右键单击该文件,选择属性,将“生成操作”属性设为“嵌入的资源”即可。这样, 在编译这个项目时,这些文件就会被编译成生成的装配件(原来 装配件这个词指的是dll等程序集)中的内嵌资源,需要注意的是,嵌入的文件名前会加上项目默认的命名空间,比如,MyNameSpace.MyJavascript.js,如果这些文件放置在项目的子目录下,那么子目录名也会成为文件名的一部分,比如,MyNameSpace.MyFolder.MyJavascript.js,如图5-5所示。

图5-5  嵌入资源

将资源嵌入装配件后,比较麻烦的问题是如何在控件中使用嵌入的资源。使用内嵌在装配件中的资源,需经过以下两个步骤
*******************************
①用System.Web.UI.WebResourceAttribute将内嵌资源标记为Web资源,WebResourceAttribute为装配件级Attribute,在使用它时需加上assembly:前缀,比如:
[assembly:WebResource(IntegrateWithJavascriptLibrary.jquery.js","text/javascript")],WebResourceAttribute可以放在assembly.cs(在VS2005中是ClassLibrary1--〉Property--〉AssemblyInfo.cs)中,也可以放在其它cs文件的namespace语句之外,而且只要一个文件声明过一次,在整个装配件中都有效。
它的第一个参数为内嵌资源文件名,后一个参数为内嵌文件的MIME类型,还可以加上第三个参数PerformSubstitution,当资源文件中使用了类似<%=WebResource (“IntegrateWithJavaScriptLibrary.tab.gif”,”image/gif”)%>这样的表达式时,则应把PerformSubstitution设为true,比如,
[assembly: WebResource("IntegrateWithJavascript Library.Tabs.css","text/css", PerformSubstitution = true)]。
使用WebResourceAttribute声明过的内嵌文件可以通过/ApplicationPath/WebResource.axd?XXX这样的路径访问。---什么意思?访问?
***********************************
②内嵌资源声明为Web资源后,使用ClientScriptManager.GetWebResourceUrl()方法获得其基于WebResource.axd的访问路径。
注:*.axd在ASP.NET 2.0网站中已被注册为使用ASP.NET解析,在根目录的Web.Config文件的httpHandlers节点中,你也可以找到
<add path="WebResource.axd" verb="GET" type="System.Web.Handlers.AssemblyResourceLoader" validate="True" />配置语句。
好了,现在我们可以完成TabbableTextArea控件了,先根据上述步骤将jquery.js和interface.js两个脚本文件包含到项目中,将设为内嵌资源,然后,将它们设为Web资源:


**************************************
     
[assembly:WebResource(   " 
  IntegrateWithJavascriptLibrary.jquery.js 
  " 
  

                        ,   " 
  text/javascript 
  " 
  )]

[assembly:WebResource(   " 
  IntegrateWithJavascriptLibrary.interface.js 
  " 
  

                        ,   " 
  text/javascript 
  " 
  )]

   namespace 
   IntegrateWithJavascriptLibrary

   {

    public class TabbableTextArea:TextBox

    {

        bool _supportJS;

        [DesignerSerializationVisibility(

                        DesignerSerializationVisibility.Hidden)]

        [Browsable(false)]

        [EditorBrowsable(EditorBrowsableState.Advanced)]

        public override TextBoxMode TextMode

      {

            get

           {   return TextBoxMode.MultiLine;    }

            set
             {

    throw new NotSupportedException(
                    "Can not change the TextMode property");
            }

        }

        void DetermineJS()
       {

            if (!DesignMode)

           {

                if (Page.Request.Browser.EcmaScriptVersion.Major > 0 

                        && Page.Request.Browser.W3CDomVersion.Major > 0)//访问本程序的浏览器支持脚本

                {
                    this._supportJS = true;

                 }

            }

        }

        protected override void OnPreRender(EventArgs e)

       {

            base.OnPreRender(e);

            DetermineJS();

            if (_supportJS)

           {

                Page.ClientScript.RegisterClientScriptResource(this.GetType(),"IntegrateWithJavascriptLibrary.jquery.js");

                Page.ClientScript.RegisterClientScriptResource(this.GetType(),"IntegrateWithJavascriptLibrary.interface.js");

                Page.ClientScript.RegisterStartupScript
 (this.GetType(),this.UniqueID,
       string.Format("    $('#{0}').EnableTabs();\r\n",
       this.UniqueID), true);

            }

        }

    }

}

代码非常简洁,这得益于组件化的客户端脚本。在重写的OnPreRender()方法中,使用ClientScriptManager.RegisterClientScriptResource()方法将Web资源注册到控件所在页面中,需要注意的是,ClientScriptManager类能自动保证同样的资源文件在同样页面中只注册一次。----是不是在同一个"方案(项目)"中也“只注册一次”。