随笔-185  评论-896  文章-4  trackbacks-64

前言

当使用MasterPage、UserControl等容器时,为了避免控件的重复命名,asp.net会自动将容器中的控件生成一个ClientID(Control Tree中的可生成,否则不会生成)。

例如:ContentPlaceHolder1中的Button1默认情况下会生成“ctl00_ContentPlaceHolder1_Button1”的ClientID。

我们在Render出来的mark up中看到的也是这些ClientID。所以,当我们使用JavaScript对控件元素进行操作的时候,必须使用ClientID来对控件进行查找。

 

Inline情况下的解决方案

如果JavaScript代码写在.aspx文件中时,也就是Inline Script时。在页面生成的时候,我们能够通过绑定机制将控件的ClientID绑定到页面Mark up中,故可使用:

document.getElementById("<%=Me.txtTest.ClientID %>" )

来获取一个控件的真实引用,当然,FindControl等方法也可以写在<%=...%>中用来绑定服务端数据到客户端。

 

external JS情况下的解决方案

然而,部分情况下,为了解耦,我们常常把JavaScript单独写在.js文件中,再引用到aspx文件中去。这种情况下,.js文件内的代码不能通过<%=...%>来进行服务端数据的绑定,所以上面的方法是不能用的。

此时简单点的解决方案就是直接在JavaScript中写控件的ClientID,但这样增加了JS文件和ASPX的耦合度,非常不推荐使用。

我常用的方法有两种,在此抛砖引玉:

 

案例:

Default5.aspx是MasterPage.master 的内容页,本例中的主要文件。

JScript.js是一个外部的js文件,用来处理JavaScript操作。

Button1是Default5.aspx中的一个<ASP:Button>,用来显示效果。

Button2是Default5.aspx中的一个<input type=button>,用来触发JavaScript。

需求:点击Button2,将Button1上的文本改成“from extended js”

 

方案一:使用内联JS访问器

要想在外部JS中获得ASPX动态生成的ClientID,可以通过在ASPX页面中添加访问器的方式来实现,类似OO语言中的属性:

我们在Default5.aspx中添加如下代码:

作用:①声明getClientId访问器,并注册Button1的ClientID。②引用JScript.js文件

<script type="text/javascript">
    
function getClientId()
    {
     
var paraId1 = '<%= Button1.ClientID %>'
;
     
return
 {Id1:paraId1};
    }
</script>

<script type="text/javascript" src="JScript.js"></script>

 

接下来,我们在JScript.js中,就可以这样来实现需求:

function ChangeText()
{
    
var btn=
document.getElementById(getClientId().Id1);
    btn.value
="from extended js"
;
}

 

getClientId().Id1 貌似很OO,而且还支持VS2008很蹩脚的JS智能提示,打上“.”之后就可以在提示中选择Id1了

如果有多个控件需要注册,只需将他们注册到访问器中即可,下面是一个完整的Demo代码:

 

Default5.aspx

 

JScript.js

 

方案二:使用JS全局变量

还有一种方法也比较OO,就是使用JS全局变量,同样,也需要在Default5.aspx中添加一段JS代码,作为全局变量,来提供ClientID:

 

<script type="text/javascript">  
    
var globals =
 {}; 
    globals.controlIdentities 
=
 {};
    globals.controlIdentities.someControl1 
= '<%= Button1.ClientID %>'
;
    globals.controlIdentities.someControl2 
= '<%= TextBox1.ClientID %>'
;
</script>

<script type="text/javascript" src="JScript.js"></script>

 

接下来,我们在JScript.js中,就可以这样来实现需求:

 

function ChangeText()

    
var btn=
document.getElementById(globals.controlIdentities.someControl1);
    btn.value
="from extended js"
;
}

 

globals.controlIdentities.someControl1,同样,也支持VS2008很蹩脚的JS智能提示,打上“.”之后就可以在提示中选择someControl1了

下面是一个完整的Demo代码:

 

Default5.aspx

 

JScript.js

 

结束语:

在上面两种方法中,也没有真正的实现aspx和js的完全解耦,所以,在js文件中,最好还是加上:

///<reference path="Default5.aspx"/>

 

上面的方法是我常用的,今天仓促的总结了一下,希望在此能够抛砖引玉,谢谢!


作者:Lance ZhangLance Zhang's Tech Blog
出处:http://blodfox777.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

posted on 2008-10-06 11:54 LanceZhang 阅读(3487) 评论(13)  编辑 收藏 网摘 所属分类: ASP.NET客户端开发常见问题

评论:
#1楼 2008-10-06 11:58 | hyf168[未注册用户]
太好啦,哈哈正在想怎么做哪?谢谢楼主
  回复  引用    
#2楼 2008-10-06 12:04 | zzzz[未注册用户]
还是要用到inline...
  回复  引用    
#3楼 2008-10-06 13:22 | Hurry      
Page.ClientScript.RegisterClientScriptBlock(typeof(Page), "ClientID", "\nvar btn1ID=" + this.btn1.ClientID+ "\n;", true);
  回复  引用  查看    
#4楼 2008-10-06 13:26 | 有容乃大      
return {Id1:paraId1};
这个写法挺有意思,我以前都采用第三种方法。

-----------------------------------------------
.net项目开发工具(V3.0 ):
http://www.cnblogs.com/mrhgw/archive/2008/08/06/1261664.html" target="_new">http://www.cnblogs.com/mrhgw/archive/2008/08/06/1261664.html
http://www.mrhgw.cn" target="_new">http://www.mrhgw.cn

  回复  引用  查看    
#5楼[楼主] 2008-10-06 13:52 | LanceZhang      
--引用--------------------------------------------------
Hurry: Page.ClientScript.RegisterClientScriptBlock(typeof(Page), &quot;ClientID&quot;, &quot;\nvar btn1ID=&quot; + this.btn1.ClientID+ &quot;\n;&quot;, true);
--------------------------------------------------------

这种方法也很主流的,但需要使用code-behind来参与,谢谢分享!

  回复  引用  查看    
#6楼 2008-10-06 15:13 | Wuya      
为什么不将Jscript.js文件中的ChangeText()函数修改类似下面的方式呢:
function ChangeText(strControlId,strText)
{
var bSuccessed=true;
try
{
var objControl=document.getElementById(strControlId);
objControl.value=strText;
}
catch
{
bSuccessed=false;
}
return bSuccessed;
}

类似于SetWindowText这个API:
This function changes the text of the specified window's title bar, if it has one. If the specified window is a control, the text of the control is changed.
BOOL SetWindowText(
HWND hWnd,
LPCTSTR lpString
);

  回复  引用  查看    
#7楼 2008-10-06 15:50 | Kai.Ma      
这样好累。
因为是webform,在js里面取控件ID很蹩脚,没有asp.net mvc那么直接

  回复  引用  查看    
#8楼[楼主] 2008-10-06 16:11 | LanceZhang      
@Wuya

Good work!

又学到了一招,呵呵,写这个例子时没考虑那么多

  回复  引用  查看    
#9楼[楼主] 2008-10-06 17:01 | LanceZhang      
@Kai.Ma

是啊,如果是ASP.NET MVC 的话,在这一方面就清爽多了

  回复  引用  查看    
#10楼[楼主] 2008-10-06 17:04 | LanceZhang      
@张小三
不妨分享一下您使用的方法啊?

  回复  引用  查看    
#11楼 2008-10-07 08:52 | 王国金      
谢谢楼主分享!
  回复  引用  查看    
#12楼 2008-10-17 09:57 | lspcieee      
个人比较喜欢这么做:
先给控件的外层标签定义一个id
然后用javascript类似这么获取:

var myTd=document.getElementById("ShoujianrenTd1");
var shoujianrenTextBox=myTd.getElementsByTagName("input")[0];

我觉得如此分离客户端和服务端代码,尽量减少耦合比较好。

  回复  引用  查看    
#13楼 2008-10-31 11:55 | 为爱走天涯      
我用的也和楼主差不多的方法。
  回复  引用  查看    
发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 1304586




相关文章:

相关链接: