添加中文字庫


要求:原操作系統代碼里只是支持了日語顯示,需要做的是實現對這個系統的漢字全角支持。

hzk16的介紹以及簡單的使用方法

HZK16字庫是符合GB2312標准的16×16點陣字庫,HZK16的GB2312-80支持的漢字有6763個,符號682個。其中一級漢字有3755個,按聲序排列,二級漢字有3008個,按偏旁部首排列。我們在一些應用場合根本用不到這么多漢字字模,所以在應用時就可以只提取部分字體作為己用。

HZK16字庫里的16×16漢字一共需要256個點來顯示,也就是說需要32個字節才能達到顯示一個普通漢字的目的。

我們知道一個GB2312漢字是由兩個字節編碼的,范圍為A1A1~FEFE。A1-A9為符號區,B0到F7為漢字區。每一個區有94個字符(注意:這只是編碼的許可范圍,不一定都有字型對應,比如符號區就有很多編碼空白區域)。下面以漢字“我”為例,介紹如何在HZK16文件中找到它對應的32個字節的字模數據。

前面說到一個漢字占兩個字節,這兩個中前一個字節為該漢字的區號,后一個字節為該字的位號。其中,每個區記錄94個漢字,位號為該字在該區中的位置。所以要找到“我”在hzk16庫中的位置就必須得到它的區碼和位碼。(為了區別使用了區碼和區號,其實是一個東西,別被我誤導了)

區碼:區號(漢字的第一個字節)-0xa0 (因為漢字編碼是從0xa0區開始的,所以文件最前面就是從0xa0區開始,要算出相對區碼)

位碼:位號(漢字的第二個字節)-0xa0

這樣我們就可以得到漢字在HZK16中的絕對偏移位置:

offset=(94*(區碼-1)+(位碼-1))*32

注解:1、區碼減1是因為數組是以0為開始而區號位號是以1為開始的

2、(94*(區號-1)+位號-1)是一個漢字字模占用的字節數

3、最后乘以32是因為漢字庫文應從該位置起的32字節信息記錄該字的字模信息(前面提到一個漢字要有32個字節顯示)

 有了偏移地址就可以從HZK16中讀取漢字編碼了

實現思路:

  1. 了解HZK編碼,理解一下符合GB2312標准的中文點陣字庫文件HZK16;
  2. 下載中文GB2312的二進制點陣文件;
  3. 將HZK16.fnt文件放入nihongo文件夾中;
  4. 修改主makefile文件和app_make.txt文件,將原來裝載nihongo.fnt的語句替換成裝載HZK16.fnt即可;
  5. 修改bootpack.c文件,將之前分配的裝載日語字體的內存擴大,載入字庫的文件名;
  6. 在haribote/graphic.c中添加支持漢字的代碼,增加一個函數用於顯示漢字;
  7. 修改putfonts8_asc函數里if (task->langmode == 3)語句塊;
  8. 測試程序。
  9. 注意:日文的編碼是分為左半部分和右半部分,而我們使用的HZK16是分為上半部分和下半部分的。

這里其他的地方比較弄,第5步將大小修改一下,我的是nihongo = (unsigned char *) memman_alloc_4k(memman, 55*94*32);

第6步,要注意,HZK16是上下兩部分,不同於日文的左右兩部分的結構。

代碼如下:

void putfont32(char *vram, int xsize, int x, int y, char c, char *font1, char *font2)
{
    int i,k,j,f;
    char *p, d ;
    j=0;
    p=vram+(y+j)*xsize+x;
    j++;
    //上半部分
    for(i=0;i<16;i++)
    {
        for(k=0;k<8;k++)
        {
            if(font1[i]&(0x80>>k))
            {
                p[k+(i%2)*8]=c;
            }
        }
        if(i%2==0){
            for(k=0;k<4;k++){
                f=p[k];
                p[k]=p[7-k];
                p[7-k]=f;
            }
        }else{
            for(k=0;k<4;k++){
                f=p[k+8];
                p[k+8]=p[15-k];
                p[15-k]=f;
            }
        }
       /* for(k=0;k<8/2;k++)
        {
            f=p[k+(i%2)*8];
            p[k+(i%2)*8]=p[8-1-k+(i%2)*8];
            p[8-1-k+(i%2)*8]=f;
        }*/
        if(i%2)
        {
            p=vram+(y+j)*xsize+x;
            j++;
        }
    }
    //下半部分
    for(i=0;i<16;i++)
    {
        for(k=0;k<8;k++)
        {
            if(font2[i]&(0x80>>k))
            {
                p[k+(i%2)*8]=c;
            }
        }
        if(i%2==0){
            for(k=0;k<4;k++){
                f=p[k];
                p[k]=p[7-k];
                p[7-k]=f;
            }
        }else{
            for(k=0;k<4;k++){
                f=p[k+8];
                p[k+8]=p[15-k];
                p[15-k]=f;
            }
        }
        /*for(k=0;k<8/2;k++)
        {
            f=p[k+(i%2)*8];
            p[k+(i%2)*8]=p[8-1-k+(i%2)*8];
            p[8-1-k+(i%2)*8]=f;
        }*/
        if(i%2)
        {
            p=vram+(y+j)*xsize+x;
            j++;
        }
    }
    return;
}

 

運行結果,我們在euc.txt中加入一些漢字。

參考資料:

1.https://www.cnblogs.com/wunaozai/p/3858473.html 30天操作系統支持中文。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM