嵌入式系統(tǒng)多語言文本解決方案
在嵌入式系統(tǒng)開發(fā)中,某些產(chǎn)品可能會需要跨區(qū)域銷售,因此,通常會有多語言的需求。對于這一類多語言需求的解決,在嵌入式產(chǎn)品中有其特殊的地方。以下,給出一種可能的解決方案。
該方案的核心思想是為所有文本建立索引,通過索引可以得到特定語言的文字編碼,隨后通過該編碼獲得字庫資源,并進行輸出。在這過程中,唯一需要注意的是對于特殊的某些語言,如阿拉伯語等的處理。(阿拉伯語字符在連寫時,其形狀會發(fā)生變化。)
1. 字庫的建立:
文本最終都將輸出給用戶,因此,必須為文本內(nèi)容指定字庫。本方案中采用UNICODE編碼字庫。字庫文件采用二進制存儲,按UNICODE編碼順序排列存儲點陣數(shù)據(jù),點陣大小為24*24。
2. 文本資源文件:
文本資源文件描述了特定語言的文本內(nèi)容,以及相關(guān)的字符編碼。例如對于Chinese.cfg文件來說,就保留了一個索引為1的文本,該文本內(nèi)容為“確認”;相應(yīng)對于English.cfg文件來說,必然會同樣有一個索引為1的文本,該文本內(nèi)容為“CONfirm”。通過對所有的文本建立索引并生成文本資源文件,就為最終的解決掃清了障礙。
文本資源文件采用二進制存儲。文件頭部16個字節(jié)為描述性信息,之后是文本映射表,緊跟映射表之后為文本的實際Unicode編碼。
3. 對文本資源文件進行描述的數(shù)據(jù)結(jié)構(gòu)
typedef STruct _txtres_fileheader {
LONG lFileType; //文件類型,0x2E434647=‘.CFG‘
LONG lVersionNum; //適用版本,0x56313032=‘V102‘
LONG lMapOffset; //偏移量,文件頭到文本映射區(qū)的偏移量
LONG lDataOffset; //偏移量,文件頭到文本數(shù)據(jù)區(qū)的偏移量
} APPteXT_FILEHEADER;
4. 文本映射表結(jié)構(gòu)
typedef struct _txtres_txtmap {
WORD wTextIndes; //當前文本的索引值
WORD wTextSize; //當前文本的Unicode編碼所占用的字節(jié)數(shù)
LONG lUnicodeOffset; //從文件頭到當前文本Unicode編碼存儲位置的偏移量
} TXTRES_TXTMAP;
5. 特殊語言(阿拉伯語等)的解決
特殊語言在連寫時可能發(fā)生變化,因此采用固定字庫可能無法解決該問題。針對這種狀況可以直接新增一個自定義字庫。以阿拉伯語為例,該字庫的處理過程如下:
a. 首先將阿拉伯的文本內(nèi)容按預(yù)定格式(例如24*24)在windows系統(tǒng)上顯示輸出,并將內(nèi)容保存為圖片格式。此時圖片中便為連寫內(nèi)容。
b. 隨后,對圖片進行分割。如按照24*24進行分割便可得到特定的24*24大小的字庫內(nèi)容。
c. 最后,將原先的UNICODE編碼轉(zhuǎn)為按照之前生成的字庫來編碼。
d. 之后在程序代碼中就可利用自定義字庫與自定義編碼來顯示阿拉伯語。
最后附上部分示例代碼。
//定義文本配置文件路徑
#define TXT_FILE_ENGLISH "config/English.cfg"
#define TXT_FILE_CHINASIM "config/ChinaSim.cfg"
#define TXT_FILE_CHINATRA "config/ChinaTra.cfg"
#define TXT_FILE_KOREAN "config/Korean.cfg"
#define TXT_FILE_JAPANESE "config/Japanese.cfg"
#define TXT_FILE_SPANISH "config/Spanish.cfg"
#define TXT_FILE_RUSSIAN "config/Russian.cfg"
#define TXT_FILE_THAI "config/Thai.cfg"
#define TXT_FILE_GERMAN "config/German.cfg"
#define TXT_FILE_FRANCE "config/France.cfg"
#define TXT_FILE_ITALY "config/Italy.cfg"
#define TXT_FILE_ARABIA "config/Arabia.cfg"
#define TXT_FILE_PORTUGAL "config/Portugal.cfg"
#define TXT_FILE_HINDI "config/Hindi.cfg"
#define TXT_FILE_TURKISH "config/Turkish.cfg"[!--empirenews.page--]
#define TXT_FILE_VIETNAM "config/Vietnam.cfg"
#define TXT_FILE_SWIDISH "config/Swedish.cfg"
#define TXT_FILE_POLISH "config/Polish.cfg"
//根據(jù)文本索引及文本語言,讀取相應(yīng)的文本配置文件,以得到該文本,成功返回有效指針
GUISTRING * GetTextResource(LONG lIndex, LONG lLanguage)
{
GUISTRING * pTxt;
APPTEXT_FILEHEADER fh;
APPTEXT_MAPPING map;
STRING strFile;
WORD * pBuf;
int fd, iOff;
//確定要讀取的配置文件
switch (lLanguage)
{
case TXT_LANG_ENGLISH:
strFile = TXT_FILE_ENGLISH;
break;
case TXT_LANG_CHINASIM:
strFile = TXT_FILE_CHINASIM;
break;
case TXT_LANG_CHINATRA:
strFile = TXT_FILE_CHINATRA;
break;
case TXT_LANG_KOREAN:
strFile = TXT_FILE_KOREAN;
break;
case TXT_LANG_JAPANESE:
strFile = TXT_FILE_JAPANESE;
break;
case TXT_LANG_SPANISH:
strFile = TXT_FILE_SPANISH;
break;
case TXT_LANG_RUSSIAN:
strFile = TXT_FILE_RUSSIAN;
break;
case TXT_LANG_THAI:
strFile = TXT_FILE_THAI;
break;
case TXT_LANG_GERMAN:
strFile = TXT_FILE_GERMAN;
break;
case TXT_LANG_FRANCE:
strFile = TXT_FILE_FRANCE;
break;
case TXT_LANG_ITALY:
strFile = TXT_FILE_ITALY;
break;
case TXT_LANG_ARABIA:
strFile = TXT_FILE_ARABIA;
break;
case TXT_LANG_PORTUGAL:
strFile = TXT_FILE_PORTUGAL;
break;
case TXT_LANG_HINDI:
strFile = TXT_FILE_HINDI;
break;
case TXT_LANG_TURKISH:
strFile = TXT_FILE_TURKISH;
break;
case TXT_LANG_VIETNAM:
strFile = TXT_FILE_VIETNAM;
break;
case TXT_LANG_SWIDISH:
strFile = TXT_FILE_SWIDISH;
break;
case TXT_LANG_POLISH:[!--empirenews.page--]
strFile = TXT_FILE_POLISH;
break;
default:
return NULL;
}
//打開配置文件并檢查其格式
if ((fd = open(strFile, O_RDONLY)) == -1)
{
return NULL;
}
if (read(fd, &fh, 16) != 16)
{
close(fd);
return NULL;
}
if (fh.lFileType != 0x4746432E || fh.lVersionNum != 0x32303156)
{
close(fd);
return NULL;
}
//在文本映射區(qū)內(nèi)查找匹配的文本索引
for (iOff = fh.lMapOffset; iOff < fh.lDataOffset; iOff += 8)
{
if (read(fd, &map, 8) != 8)
{
close(fd);
return NULL;
}
if (map.wTextIndex == lIndex)
{
break;
}
}
if (iOff >= fh.lDataOffset)
{
close(fd);
return NULL;
}
//根據(jù)找到的文本映射來讀取文本內(nèi)容
if (!(pBuf = GuiMemAlloc(map.wTextSize + 2)))
{
close(fd);
return NULL;
}
lseek(fd, fh.lDataOffset + map.lTextOffset, SEEK_SET);
if (read(fd, pBuf, map.wTextSize) != map.wTextSize)
{
GuiMemFree(pBuf);
close(fd);
return NULL;
}
pBuf[map.wTextSize >> 1] = 0;
//建立字符串對象
pTxt = CreateStringDirect(pBuf);
GuiMemFree(pBuf);
close(fd);
return pTxt;
}
//定義與字符串相關(guān)的數(shù)據(jù)結(jié)構(gòu)
#ifndef GUI_STRING_STRUCT
typedef struct _string
{
WORD wWidth; //字符串寬度,字符串輸出時的總點陣寬度
WORD wLength; //字符串長度,不包括‘