最近在做一个小程序,要求实现对多语言界面显示支持功能,并且,界面显示内容用户能够自己设置。
初步设计用INI文件来配置显示内容,换一种语言的配置文件,就能够更换整个系统的显示语言。考虑到系统规模很小,周期又短,不想用太复杂的方案来解决这个问题,当参考了很多网上类似的设计和代码,发现都不是很满足。
主要问题在于绝大多数基于INI文件配置这种简单应有实现的代码,都是针对组件ID固定加载,写死了组件的ID号,比如 以下是引用片段
strCaption = fileManager.GetString(section,"IDC_Stc_ListStudent","");
SetDlgItemText(IDC_Stc_ListStudent,strCaption);
strCaption = fileManager.GetString(section,"IDC_Stc_AllContent","");
SetDlgItemText(IDC_Stc_AllContent,strCaption);
这样界面组件越多,加载代码越长;每新增一个显示窗口,又必须复制、粘贴类似的代码,根据组件ID常量值来修改相关的加载项,很是不爽!
初步设想是设计统一、通用的窗口组件Caption设置方法,对给定的Frame或Dialog等Window容器组件内的所以组件进行遍历,当增、减显示组件不对语言包加载代码产生影响,达到自适应界面组件语言包加载效果。
这样就产生一个新问题语言包配置文件中的Caption值如何跟相关的组件正确地一一对应?
好友文国庆建议用XML文件来定义这种对应关系。这个想法触动了我反正就是一个[Key,Value]的数据,就用已经实现的INI配置文件也可以啊。于是所有问题解决!
具体设计是语言包配置文件就直接设置成组件ID与组件显示信息的Hash表,Key = Value的形式,比如BTnOK组件的ControlID为“1003”,中文显示Caption为“登录”,语言包配置内容就是“1003=登录”。
语言包的加载过程为2步实现
首先,从语言包配置文件中,读取所有配置的ID、Caption条目到Vector或者Array中。
其次,在遍历指定窗口中所有组件时,每发现一个组件,就用其ID在已经加载的语言包数组中查找,找到就用配置的值修改组件Caption属性;找不到,就认为是不需要动态配置,不做处理。
配置文件实例
配置项解释Section[Login Dialog]界面窗口;等号左边窗口中需要设置其Caption属性的组件ID;等号左边窗口中需要设置其Caption属性的组件Caption值;
[Login Dialog]
1001 = 用户帐号
1002 = 用户密码
1017 = 登 录
1018 = 退 出
语言包配置信息加载代码
以下是引用片段
BOOLCLanguageManager::loadFromFile()
...{
BOOLbRead=FALSE;
inti;
ItemContexttemp;
CStringArrayitemBuf,valueBuf;
bRead=fileManager.GetSectionValues("MainWindow",itemBuf,valueBuf);
if(bRead)
...{
for(i=0;i
...{
temp.UCtrlID=atoi(itemBuf.GetAt(i));
temp.strContext=valueBuf.GetAt(i);
m_vtContexts.push_back(temp);
}
}
itemBuf.RemoveAll();
valueBuf.RemoveAll();
bRead=fileManager.GetSectionValues("LoginDialog",itemBuf,valueBuf);
if(bRead)
...{
for(i=0;i
...{
temp.uCtrlID=atoi(itemBuf.GetAt(i));
temp.strContext=valueBuf.GetAt(i);
m_vtContexts.push_back(temp);
}
}
returnbRead;
}
读取语言包配置信息
以下是引用片段
BOOLCIniFile::GetSectionValues(CStringSection,CStringArray%26amp;strItemBuf,CStringArray%26amp;strValueBuf)
...{
BOOLbRead=FALSE;
ReadIniFile();//打开文件
if(bFileExist==FALSEFileContainer.GetSize()%26lt;0)
returnbRead;//文件打开出错或文件为空,返回默认值
inti=0;
intiFileLines=FileContainer.GetSize();
CStringstrline,str;
while(i
...{
strline=FileContainer.GetAt(i++);
strline.TrimLeft();
if(strline.GetLength()%26lt;=0)
continue;//跳过空行
if(strline.Left(2)=="//")
continue;//跳过注释行
if(strline.GetAt(0)=='[')//查找Section,第一个必须为[
...{
str=strline.Left(strline.Find("]"));//去掉]右边
str=str.Right(str.GetLength()-str.Find("[")-1);//去掉[左边
str.TrimLeft();
str.TrimRight();
if(Section==str)//找到Section
...{
while(i
...{
strline=FileContainer.GetAt(i++);
strline.TrimLeft();
if(strline.GetLength()%26lt;=0)
continue;//跳过空行
if(strline.GetAt(0)=='[')
returnbRead;//假如到达下一个[],即找不到,返回默认值
if(strline.Left(2)=="//")
continue;//跳过注释行
str=strline.Left(strline.Find("="));//去掉=右边
str.TrimLeft();
str.TrimRight();
//保存等号左边项
strItemBuf.Add(str);
str=strline.Right(strline.GetLength()-strline.Find("=")-1);//去掉=左边
str.TrimLeft();
str.TrimRight();
//保存等号右边项
strValueBuf.Add(str);
bRead=TRUE;
}
//当前Section遍历结束
}
//没有找到Section
}
//当前行遍历结束
}
returnbRead;
}
修改指定组件Caption属性代码
以下是引用片段
BOOLCLanguageManager::setControlCaption(CWnd*pCtrl,UINTctrlID)
...{
BOOLisOK=FALSE;
for(inti=0;i
...{
isOK=(m_vtContexts[i].uCtrlID==ctrlID);
if(isOK)
...{
pCtrl-%26gt;SetWindowText(m_vtContexts[i].strContext);
break;
}
}
returnisOK;
}
遍历设置指定窗口所有组件Caption属性代码
以下是引用片段
voidCLanguageManager::setCaptionForWindow(CWnd*pWnd)
...{
//枚举对话框中所有组件
CWnd*pCtrl=pWnd-%26gt;GetWindow(GW_CHILD);
while(pCtrl!=NULL)
...{
UINTctrlID=pCtrl-%26gt;GetDlgCtrlID();
setControlCaption(pCtrl,ctrlID);
pCtrl=pCtrl-%26gt;GetNextWindow();
}
}
