当使用MasterPage、UserControl等容器时,为了避免控件的重复命名,asp.net会自动将容器中的控件生成一个ClientID(Control Tree中的可生成,否则不会生成)。
例如:ContentPlaceHolder1中的Button1默认情况下会生成“ctl00_ContentPlaceHolder1_Button1”的ClientID。
我们在生成的HTML代码中看到的也是这些ClientID。所以,当我们使用JavaScript对控件元素进行操作的时候,必须使用ClientID来对控件进行查找。
一、当JavaScript脚本写在.aspx页面中时:
如果JavaScript代码写在.aspx文件中时,在页面生成的时候,我们能够通过绑定机制将控件的ClientID绑定到页面中,故可使用:
document.getElementById("<%=txtTest.ClientID %>" )
如果是Jquery就是:$("#<%=txtTest.ClientID %>")
来获取ID为txtTest的服务端控件,当然,FindControl等方法也可以写在<%=...%>中用来绑定服务端数据到客户端。
一、当JavaScript脚本写在外部.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 %>'
;
//注册控件1
var
paraId2 =
'<%= TextBox1.ClientID %>'
;
//注册控件2
var paraId3='<%=Lable1.ClientID%>'; //注册控件3
return
{Id1:paraId1,Id2:paraId2,Id3:paraId3};
//生成访问器
}
</script>
<script type=
"text/javascript"
src=
"JScript.js"
></script>
|
接下来,我们在JScript.js中,就可以这样来实现需求:
function
ChangeText() {
var
btn=document.getElementById(getClientId().Id1);
btn.value=
"from extended js"
;
var lable1=document.getElementById(getClientId().Id3);
lable1.innerHTML="FROM Extened Js";
}
|
getClientId().Id1 貌似很OO,而且还支持VS2008很蹩脚的JS智能提示,打上“.”之后就可以在提示中选择Id1了
如果有多个控件需要注册,只需将他们注册到访问器中即可,下面是一个完整的Demo代码:
Default5.aspx代码:
<%@ Page Language=
"C#"
MasterPageFile=
"~/MasterPage.master"
Title=
"Untitled Page"
%>
<script runat=
"server"
>
</script>
<asp:Content ID=
"Content1"
ContentPlaceHolderID=
"head"
Runat=
"Server"
>
<script type=
"text/javascript"
>
function getClientId() {
var paraId1 =
'<%= Button1.ClientID %>'
;
//注册控件1
var paraId2 =
'<%= TextBox1.ClientID %>'
;
//注册控件2
var paraId3='<%=Lable1.ClientID%>'; //注册控件3
return
{Id1:paraId1,Id2:paraId2,Id3:paraId3};
//生成访问器
}
</script>
<script type=
"text/javascript"
src=
"JScript.js"
></script>
//引用外部js
</asp:Content>
<asp:Content ID=
"Content2"
ContentPlaceHolderID=
"ContentPlaceHolder1"
Runat=
"Server"
>
<asp:TextBox ID=
"TextBox1"
runat=
"server"
></asp:TextBox>
<asp:Button ID=
"Button1"
runat=
"server"
Text=
"Button"
/>
<asp:Lable ID="Lable1" runat=
<input id=
"Button2"
type=
"button"
value=
"button"
onclick=
"ChangeText();"
/>
</asp:Content>
|
JScript.js代码:
function
ChangeText() {
var
btn=document.getElementById(getClientId().Id1);
btn.value=
"from extended js"
;
var
btn=document.getElementById(getClientId().Id2);
btn.value=
"from extended js"
;
var lable1=document.getElementById(getClientId().Id3);
lable1.innerHTML="FROM Extened 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代码:
<%@ Page Language=
"C#"
MasterPageFile=
"~/MasterPage.master"
Title=
"Untitled Page"
%>
<script runat=
"server"
>
</script>
<asp:Content ID=
"Content1"
ContentPlaceHolderID=
"head"
Runat=
"Server"
>
<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>
</asp:Content>
<asp:Content ID=
"Content2"
ContentPlaceHolderID=
"ContentPlaceHolder1"
Runat=
"Server"
>
<asp:TextBox ID=
"TextBox1"
runat=
"server"
></asp:TextBox>
<asp:Button ID=
"Button1"
runat=
"server"
Text=
"Button"
/>
<input id=
"Button2"
type=
"button"
value=
"button"
onclick=
"ChangeText();"
/>
</asp:Content>
|
JScript.js代码:
function
ChangeText()
{
var
btn=document.getElementById(globals.controlIdentities.someControl1);
btn.value=
"from extended js"
;
var
txt=document.getElementById(globals.controlIdentities.someControl2);
btn.value=
"from extended js"
;
}
|
结束语:
在上面两种方法中,也没有真正的实现aspx和js的完全解耦,所以,在js文件中,最好还是加上:
///<reference path="Default5.aspx"/>
这样就可以在外部JS文件中得到智能提示