在ASP.Net 2.0 中提供多語言轉換和多樣式主題轉換功能,兩種實現形式比較類似,所以放在一起說明一下。
1. Language switcher 多語言轉換
在Quick Start Tutorial 中,介紹了如何存儲和應用用戶選擇的語言。一般是用一個DropDownList展示支持的語言,供用戶選擇,通常是放在masterpage 裡面,將用戶選擇的語言存儲起來 這裡用了ASP.Net 2.0的Profile,當然也可以存在cookIE session 或者querystring裡。在頁面裡重寫InitializeCulture 方法,使用用戶之前選擇的語言。因為設置語言的操作 (這裡是SelectedIndExchanged事件)發生在InitializeCulture 時間後面,所以在設置操作完成後為了使的當前頁面也馬上生效,需要做個重轉向,以從新加載本頁面,觸發InitializeCulture 事件。下面使quickstart中的部分代碼,注意紅色部分。因為有的頁面地址後面可能還存在queystring,所以個人覺得紅色代碼部分最好用Response.Redirect(Request.Url.PathAndQuery);代替。
protected void DropDownLanguage_SelectedIndExchanged(object sender, EventArgs e)
{
string SelectedLanguage = DropDownLanguage.SelectedValue.ToString();
//Save selected user language in profile
Profile.SetPropertyValue("PreferredCulture", SelectedLanguage);
//Force re-initialization of the page to fire InitializeCulture()
Response.Redirect(Request.Url.LocalPath);
}
protected override void InitializeCulture()
{
// override virtual method InitializeCulture() to check if profile contains a user language setting
string UserCulture = Profile.GetPropertyValue("PreferredCulture").ToString();
if ( UserCulture != "")
{
// there is a user language setting in the profile: switch to it
Thread.CurrentThread.CurrentUICulture = new CultureInfo(UserCulture);
Thread.CurrentThread.CurrentCulture = CultureInfo.CreateSpecificCulture(UserCulture);
}
}
為了減少代碼的重復,一般會自定義一個customer base page類,使它繼承Page類,然後在自定義的頁基類中重新InitializeCulture方法。最後把你的每個頁面繼承自你的自定義頁面基類。這樣你就不需要每個頁面都重寫InitializeCulture方法了。
但是上面這個方法還是不是很爽,因為每添加一個頁面都要去修改後置代碼,來繼承自定義頁基類。
我們注意到,在InitializeCulture方法中實際上只是修改了當前線程的Culture和UICulture。那麼可不可以在一個全局的事件中,比如Application的某個事件,來修改這兩個屬性呢?很早以前我這麼試過,在Application的BeginRequest事件觸發時來實現InitializeCulture 的細節,類似於下面代碼:
void Application_BeginRequest(object sender, EventArgs e)
{
string lang = string.Empty;//default to the invariant culture
lang = Profile.PreferredCulture;
if (string.IsNullOrEmpty(lang))
{
&
nbsp; lang = string.Empty;
}
Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo(lang);
Thread.CurrentThread.CurrentCulture = CultureInfo.CreateSpecificCulture(lang);
}
注意紅色部分應用其他方式取代,因為在beginrequest觸發階段,profile對象還沒有被ASP.Net創建。可以用cookIEs取代。
我記得當時這麼做後,語言設置後並不起作用,當時認為在全局事件中處理,可能到後來還是會被覆蓋掉,所以可能不行。所以當時還是用了 InitializeCulture方法。今天在ASP.Net論壇裡看到有人如此實現了,
void Application_BeginRequest(Object sender, EventArgs e){
string lang = string.Empty;//default to the invariant culture
HttpCookie cookie = Request.CookIEs["DropDownName"];
if (cookie != null && cookIE.Value != null)
lang = Request.Form[cookIE.Value];
Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo(lang);
Thread.CurrentThread.CurrentCulture = CultureInfo.CreateSpecificCulture(lang);
}
所以覺得當時可能哪裡沒有設置好,於是又試了一次,原來是頁面頭指令<%@ Page UICulture="auto" Culture="auto" %>的原因,如果在頁面中設置了UICulture和Culture後,它們就會覆蓋掉在全局中的設置。去掉之後,全局設置起作用了。看來頁面中的culture的設置會覆蓋全局的設置,而頁面中InitializeCulture方法(確切說是一切支持該方法的控件)的設置會覆蓋頁面的設置。其實在Page類中InitializeCulture方法的默認實現是空的,因此再將頁面頭指令 UICulture="auto" Culture="auto" 去掉後,Global中的設置就起作用了。
另外,如果很想使用Profile(像我一樣)來存儲用戶的選擇,那就不能在beginrequest階段來處理了,我是在PreRequestHandlerExecute事件觸發時處理:
void Application_PreRequestHandlerExecute(object sender, EventArgs e)
{
string lang = string.Empty;//default to the invariant culture
lang = Profile.PreferredCulture;
if (string.IsNullOrEmpty(lang))
{
lang = string.Empty;
}
Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo(lang);
Thread.CurrentThread.CurrentCulture = CultureInfo.CreateSpecificCulture(lang);
}
這個時候Profile已經被創建了,所以可以使用了。
2. 多樣式主題轉換 Theme switcher
這篇文章講了Theme的切換,覺得形式上和語言的切換很類似。他使用了HttpModule,我覺得直接放在Global.asax文件裡對應的事件處理發放下就可以了,說到底都是一樣的。他的存儲采用了cookIE,我還時覺得用Profile好,既然提供了就用呗,Profile應該是有緩存的吧,所以性能應該不是問題。
出處:厚積而勃發 BLOG