4 前端界面模块实现
4.1 窗口的创建
4.1.1 创建一个窗口的过程
输入法的前端界面里面,包括状态窗口,编码输入窗口与候选窗口三个主要的窗口。这些窗口都需要通过Windows程序的API所提供我们一系列的函数创建。创建一个窗口的过程如下:
- 初始化WinMain 函数
2.定义和注册窗口类
3.利用CreateWindow函数创建一个窗口
4.创建窗口函数后,调用ShowWindow 函数显示窗口,以及使用UpdateWindow函数更新窗口。
5.窗口进入消息循环
综述, 一个窗口的创建需要经过初始化WinMain入口函数 ,定义注册窗口类,
创建窗口,显示和更新窗口,最后到消息循环这一个过程。
4.2 窗口贴图
4.2.1 为什么要进行窗口贴图
输入法至今都有好一段的历史,在这段时间里面。输入法无论在外观和功能上都发生了巨大的改变。特别在用户界面上,增加很多的设计元素在内,窗口的外观上有了质的飞跃。话说回来,看到标题也许会问:“同学你在说什么呢?说那么一堆无用的话”。其实要讲的是关于一些外观上的事情,联想到Windows为什么能够取得那么多用户的青睐,很大程度是在于它美化的图形外观,以及方便简单的操作。话说到这里,我引出的事情是就是关于输入法窗口的贴图。什么是窗口贴图?它的意义所在呢?
所谓的窗口贴图就是在程序所创建的窗口基础上,进行外观上的改造,美化等操作。它的意义其中有:
第一,图形用户界面能够为用户提供界面友好的操作环境。
第二,图形用户界面能够吸引用户的注意力和关注。
第三,图形用户界面能为软件增添内涵。
以上是从功能上,外观上,价值上讲述用户界面的作用。就是因为这些作用,我们有理由相信为了让用户得到更好的体现,美化窗口是必要的而且是根据历史的发展经验告诉我们的,丰富程序的用户界能为用户带来更多,更好的体现。
4.2.2 窗口贴图的原理
在进行贴图之前,我们需要需要了解DDB位图和DIB位图之前的相关的事情。使用位图资源时候,调用LoadBitmap函数将位图资源调入内存,此函数忽略了调色板和其他信息,使用系统调色板显示位图,这就是设备相关的位图,即DDB位图。如果保留位图的颜色表信息,颜色表描述了像素点值是如何与RGB值相对应的。这个颜色表不一定与特定的图形输出设备兼容,因此位图就称与设备无关的位图,即DIB格式位图。LoadBitmap函数具有将DIB位图转换DDB位图的功能,但只是对以资源形式存放的位图有此功能,对任意存放在磁盘上的位图,可以使用其他的相关的函数进行转换。
DDB依赖于具体设备的,DDB的颜色模式必须与输出设备相一致。例如当前的显示设备是256色模式,那么DDB必然也是256色的。DDB依依赖输出设备,所以DDB只能存在于内存中。DIB与设备无关性,在于它的颜色模式与设备无关。例如一个256色的DIB可以在真彩色显示模式下使用,同样也可以在16色模式下使用。256色以下,包括256色的DIB拥有自己的颜色表,像素独立于系统调色板,它可以永久存放在磁盘中。
根据以上的DDB与DIB上的区别,我们很容易知道其原理所在。当我们把已经制作好的256色的BMP位图导入Visual c++ 里面作为一种系统资源使用,保存在磁盘上,当程序需要这个位图的时候,那么就将其DIB位图转换DDB位图,然后通过相关使用DDB位图的函数显示到指定的窗口。
图4-1 DIB与DDB图示
4.2.3 窗口贴图的过程
窗口贴图大概一个过程如下图:
4-1 图为窗口贴图流程图
接下来的工作就是要了解这样的一个贴图代码实现过程,大概有一个感性认识的过程。在窗口中显示图像时的过程:
一、首先是获取设备环境中的句柄。
二、获取与设备环境兼容的内存设备描述表,将要显示位图的句柄选入内存设备描述表,即先将位图显示在内存中。
三、将内存设备描述表中的位图通过BitBlt函数显示在窗口屏幕中。
四、删除内存设备环境,释放资源
当应用程序运行时需要调用到的资源才被装入到内存里面,当不需要调入到内存里面,资源仍然保留在磁盘上。
4.2.4 窗口贴图知识点
1.位图资源
在进行贴图之前,我们还是有必要先了解一下有关于位图资源的知识。位图(BITMAP)是由点阵数据表示的图像。位图在Windows中 主要有两个用途:一、在屏幕上显示图像。在Windows系统中包含了很多的位图,有滚动条的箭头,系统菜单框、各种按钮等。也可以在屏幕上显示画笔创建的图像。二、创建刷子。刷子是Windows用来填充显示的像素图案。
打开Visual C++ 菜单 Insert-àResource 选择Bitmap,使用New命令进行创建位图资源。
图4-1 位图资源对话框
建立后,我们从外部加载一张256色或者少与256色的BMP位图,进行贴图。
位图资源的内存选项是可移动的,但是却是不可丢弃的。同时位图是GDI对象,在程序之间不可共享,并且当程序结束时,不能从内存自动删除。当程序结束时,使用DeleteObject函数可以将位图和刷子从内存中删除。
使用LoadBitmap函数获取位图的句柄,LoadBitmap函数的一般形式如下:
HBITMAP loadBitmap(hinst,lpszBitmap)
HINSTANCE hinst; // 实例句柄
LPCSTR lpszBitmap;//位图的名称
此函数将位图资源装入内存。
DeleteOject函数的一般形式如下:
BOOL DeleteOject(hgdiobj)
HGDIOBJ hgdiobj;// 要删除的对象的句柄
此函数从内存中删除对象,并释放所占存储空间。
2.内存设备描述表:
了解了位图资源后,我们可以利用LoadBitmap 函数获取我们所需的位图句柄,但我们发现只是获取位图句柄还不能完成我们贴图的工作。因此,我们还要了解内存设备描述表相关的知识,内存描述表是一个重要结构体,封装windows一些重要数据。我们可以比喻内存设备描述表为一个衣柜,把位图比喻为一件衣服,衣柜被划分很多格,每一个格当中只是摆放一件衣服,衣服就摆放在衣柜里面。然而衣柜能够很好管理衣服,但是没有打开衣柜之前,衣服是不可以显示的。也就是说,内存设备描述表类似一个管理位图的一个衣柜,但这个衣柜仍然是没有打开之前是不可见里面的。
3.显示DDB位图:
当位图被加载到内存后,我们并不能在我们创建的窗口显示出其位图到屏幕上。那么,为了可以将内存里的位图显示到屏幕上去,我们可以使用BitBit函数它的一般形式如下:
BOOL BitBlt(HDC hdcDest, int nXDest, int nYDest, int nWidth, int nHeight, HDC hdcSrc, int nXSrc, int nYSrc, DWORD dwRop);
HDC hdcDest; // 目的设备表描述句柄
int nXDest;// 目的矩形左上角的x坐标
int nYDest;// 目的矩形左上角的y坐标
int nWidth;// 位图的宽度
int nHeight;//位图的高度
HDC hdcSrc;//源设备描述表句柄
int nXSrc;//源位图的左上角的x坐标
int nYSrc;//源位图的左上角的y坐标
DWORD dwRop;//拷贝时的像素操作
通过这个函数,内存的位图就能够显示到屏幕当中。可以根据需要进行选取自己所需要的区域组合成自己想要的图像。
4.2.5 实现皮肤变换
为了实现输入法在外观上的变换,实现皮肤的变换,为了实现这样一个功能,需要对其进行定义一个函数来对其进行皮肤的变换。
大概的思路:
创建一个LoadSkin这个自定义函数,让它回返一个HBITMAP句柄的值。其次要利用LoadImage函数加载外部的图片,第三利用DrawState 显示加载的图片。
实现过程如下:
//加载外部图片,进行皮肤贴图,
HBITMAP LoadSkin(HWND hWnd,LPTSTR SkinName)
{
HDC hDc;
HBITMAP hBitmap ;
hDc=GetDC(hWnd);//获取设备环境DC
hBitmap=(HBITMAP)LoadImage(hInst,SkinName,IMAGE_BITMAP,0,0, LR_CREATEDIBSECTION | LR_DEFAULTSIZE |LR_LOADFROMFILE );
DrawState(hDc,NULL,NULL,(LPARAM)hBitmap,0,0,0,0,0,DST_BITMAP);
return hBitmap;
}
函数说明:
函数原型:HANDLE LoadImage(
NINSTANCE hinst,//指向要装载的图像所在的模块实例
LPCTSTR lpszName,//指向图像的文件名或其ID号
UINT uType,//指定其图像类型,包括位图,光标,图标
int cxDesired,//指定图标或者光标的宽度
int CyDesired,// 指定图标或者光标的高度
UINT fuLoad);//Load flags
BOOL DrawState(
HDC hdc, // handle to device context
HBRUSH hbr, // handle to brush
DRAWSTATEPROC lpOutputFunc, // callback function
LPARAM lData, // image information
WPARAM wData, // more image information
int x, // horizontal location
int y, // vertical location
int cx, // image width
int cy, // image height
UINT fuFlags // image type and state
);
在实现的过程种,我们只要调用其LoadSkin函数,就要实现简单变换效果。
例如:
static HBITMAP g_hBitmap;//全局换皮肤
Switch(message)
{
case WM_CREATE:
g_hBitmap=LoadSkin(hWnd,"input.bmp");
break;
……………………
…………………….
省略
……………………
}
这个过程,我们只需要指定要变换的窗口的句柄,以及外部被加载的图片名称
这样就可以实现一个简单的变换皮肤功能。