如果不能在客户端进行自定义的话, Profile Service的自定义能力还是远远不够的。虽然Profile Service没有提供一种“官方”的客户端自定义支持,不过事实上“自定义”能力“天然”地存在与客户端里。为什么?因为整个客户端是由 JavaScript实现的,这种灵活的语言使得我们能够在一定程度上自由地修改客户端的行为。一般来说,在客户端扩展Profile Service主要有两种方法:
一、使用自定义类替换Sys.Services.ProfileService对象
一般来说,这是最容易想到的办法。我们可以写一个类继承Sys.Services._ProfileService类(这个类完全通过 prototype扩展,因此对于继承非常友好),甚至完全重写一个类,这个一般就看具体情况了。假设我们已经定义了这么一个类 “Jeffz.Services.ProfileService”,并将其包含在MyProfile.Service.js中,就要开始使用了。那么还要 注意些什么呢?
需要注意的就是顺序,我们一般会使用ScriptManager引入该JS,如下:
我们为ProfileService节点加上了LoadProperties属性,表明需要预加载Profile中的ZipCode和 Address这个Profile Group下的City属性。另外,我们将EnablePartialRendering属性设为了False,避免出现多余的代码。于是,生成如下的代 码如下:
第一行引入的是MicrosoftAjax.js,它之中定义了ASP.NET AJAX中默认的ProfileService,而紧接着就是对于ProfileService的使用:设定其Path以及预加载的 Properties。在引入之后千万不能忘了要将这些信息进行保留。但是这两者之间无法插入任何代码,因此我们可以在 MyProfileService.js里添加如下的代码,以保留这些信息:
当然,可能代码会根据实际情况略有不同,但是注意JavaScript引入以及执行的顺序,在做任何自定义工作时都是非常重要的。
有人也许会问,既然已经重新定义了自己的实现,为什么还要将其“伪装”成默认的ProfileService呢?因为这种“自定义”其实并不为“官方” 所承认,这么做能够保证了兼容性,保证了第三方的组件也能使用Profile Service,即使它们没有“意识”到没有使用ASP.NET AJAX提供的默认Profile Service。
二、直接修改Sys.Services._ProfileService.prototype
Sys.Services._ProfileService完全通过prototype定义,这不光非常有利于“继承”,它的一个很大的特点就是能够通 过直接修改prototype而做到修改类的行为,它对于已经实例化的对象依然有用。例如之需要如下的代码就能够为它的load函数提供一个trace功 能:
一、使用自定义类替换Sys.Services.ProfileService对象
一般来说,这是最容易想到的办法。我们可以写一个类继承Sys.Services._ProfileService类(这个类完全通过 prototype扩展,因此对于继承非常友好),甚至完全重写一个类,这个一般就看具体情况了。假设我们已经定义了这么一个类 “Jeffz.Services.ProfileService”,并将其包含在MyProfile.Service.js中,就要开始使用了。那么还要 注意些什么呢?
需要注意的就是顺序,我们一般会使用ScriptManager引入该JS,如下:
<
asp:ScriptManager
ID
="ScriptManager1"
runat
="server"
EnablePartialRendering="false"
>
< Scripts >
< asp:ScriptReference Path ="MyProfileService.js" />
</ Scripts >
< ProfileService LoadProperties ="ZipCode, Address.City" Path="MyProfile.asmx" />
</ asp:ScriptManager >
< Scripts >
< asp:ScriptReference Path ="MyProfileService.js" />
</ Scripts >
< ProfileService LoadProperties ="ZipCode, Address.City" Path="MyProfile.asmx" />
</ asp:ScriptManager >
我们为ProfileService节点加上了LoadProperties属性,表明需要预加载Profile中的ZipCode和 Address这个Profile Group下的City属性。另外,我们将EnablePartialRendering属性设为了False,避免出现多余的代码。于是,生成如下的代 码如下:
<
script
src
="/Value-Add-WebSite/WebResource.axd?d=...;t=..."
type
="text/javascript"
></
script
>
< script type ="text/javascript" >
<!--
Sys.Services.ProfileService.set_path('/MyProfile.asmx');
Sys.Services.ProfileService.properties = Sys.Serialization.JavaScriptSerializer.deserialize('{\ " ZipCode\ " :\ " \ " }');
Sys.Services.ProfileService.properties.Address = new Sys.Services.ProfileGroup(Sys.Serialization.JavaScriptSerializer.deserialize('{\ " City\ " :\ " \ " }'));
Sys.Services.AuthenticationService._set_authenticated( true );
// -->
</ script >
< script src ="MyProfileService.js" type ="text/javascript" ></ script >
< script type ="text/javascript" >
<!--
Sys.Services.ProfileService.set_path('/MyProfile.asmx');
Sys.Services.ProfileService.properties = Sys.Serialization.JavaScriptSerializer.deserialize('{\ " ZipCode\ " :\ " \ " }');
Sys.Services.ProfileService.properties.Address = new Sys.Services.ProfileGroup(Sys.Serialization.JavaScriptSerializer.deserialize('{\ " City\ " :\ " \ " }'));
Sys.Services.AuthenticationService._set_authenticated( true );
// -->
</ script >
< script src ="MyProfileService.js" type ="text/javascript" ></ script >
第一行引入的是MicrosoftAjax.js,它之中定义了ASP.NET AJAX中默认的ProfileService,而紧接着就是对于ProfileService的使用:设定其Path以及预加载的 Properties。在引入之后千万不能忘了要将这些信息进行保留。但是这两者之间无法插入任何代码,因此我们可以在 MyProfileService.js里添加如下的代码,以保留这些信息:
var
path
=
Sys.Services.ProfileService.get_path();
if ( ! path)
{
path = Sys.Services._ProfileService.WebServicePath;
}
var properties = Sys.Services.ProfileService.properties;
var newInstance = new Jeffz.Services.ProfileService();
newInstance.set_path(path);
newInstance.properties = properties;
Sys.Services.ProfileService = newInstance;
if ( ! path)
{
path = Sys.Services._ProfileService.WebServicePath;
}
var properties = Sys.Services.ProfileService.properties;
var newInstance = new Jeffz.Services.ProfileService();
newInstance.set_path(path);
newInstance.properties = properties;
Sys.Services.ProfileService = newInstance;
当然,可能代码会根据实际情况略有不同,但是注意JavaScript引入以及执行的顺序,在做任何自定义工作时都是非常重要的。
有人也许会问,既然已经重新定义了自己的实现,为什么还要将其“伪装”成默认的ProfileService呢?因为这种“自定义”其实并不为“官方” 所承认,这么做能够保证了兼容性,保证了第三方的组件也能使用Profile Service,即使它们没有“意识”到没有使用ASP.NET AJAX提供的默认Profile Service。
二、直接修改Sys.Services._ProfileService.prototype
Sys.Services._ProfileService完全通过prototype定义,这不光非常有利于“继承”,它的一个很大的特点就是能够通 过直接修改prototype而做到修改类的行为,它对于已经实例化的对象依然有用。例如之需要如下的代码就能够为它的load函数提供一个trace功 能:
Sys.Service._ProfileService.prototype._originalLoad
=
Sys.Service._ProfileService.prototype.load;
Sys.Service._ProfileService.prototype.load = function (propertyNames, loadCompletedCallback, failedCallback, userContext)
{
for ( var i = 0 ; i < propertyNames.length; i ++ )
{
debug.trace(propertyNames[i]);
}
Sys.Service._ProfileService.prototype._originalLoad(propertyNames, loadCompletedCallback, failedCallback, userContext);
}
Sys.Service._ProfileService.prototype.load = function (propertyNames, loadCompletedCallback, failedCallback, userContext)
{
for ( var i = 0 ; i < propertyNames.length; i ++ )
{
debug.trace(propertyNames[i]);
}
Sys.Service._ProfileService.prototype._originalLoad(propertyNames, loadCompletedCallback, failedCallback, userContext);
}
事实上,如果需要修改类库已有的行为,最方便的就是这种做法。从这里可以看出JavaScript的灵活性,以及使用prototype的好处。
本文转自 jeffz 51CTO博客,原文链接:http://blog.51cto.com/jeffz/60665,如需转载请自行联系原作者