📄 18. metafile.txt
字号:
CloseClipboard () ;
如果程序使用增强型MetaFile就可以省去这项工作。当某个应用程序将这些格式放入剪贴簿而另一个程序却要求从剪贴簿中获得其它格式时,Windows剪贴簿会自动在老式MetaFile和增强型MetaFile之间进行格式转换。
增强型 MetaFile
「增强型MetaFile」格式是在32位Windows版本中发表的。它包含一组新的函数呼叫、一对新的数据结构、新的剪贴簿格式和新的文件扩展名.EMF。
这种新的MetaFile格式最重要的改进是加入可通过函数呼叫取得的更丰富的表头信息,这种表头信息可用来帮助应用程序显示MetaFile图像。
有些增强型MetaFile函数使您能够在增强型MetaFile(EMF)格式和老式MetaFile格式(也称作Windows MetaFile(WMF)格式)之间来回转换。当然,这种转换很可能遇到麻烦,因为老式MetaFile格式并不支持某些,例如GDI绘图路径等,新的32位图形功能。
基本程序
程序18-2所示的EMF1建立并显示增强型MetaFile。
程序18-2 EMF1
EMF1.C
/*----------------------------------------------------------------------------
EMF1.C -- Enhanced MetaFile Demo #1
(c) Charles Petzold, 1998
-----------------------------------------------------------------------------*/
#include <windows.h>
LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;
int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
LPSTR lpszCmdLine, int nCmdShow)
{
static TCHAR szAppName[] = TEXT ("EMF1") ;
HWND hwnd ;
MSG msg ;
WNDCLASS wndclass ;
wndclass.style = CS_HREDRAW | CS_VREDRAW ;
wndclass.lpfnWndProc = WndProc ;
wndclass.cbClsExtra = 0 ;
wndclass.cbWndExtra = 0 ;
wndclass.hInstance = hInstance ;
wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ;
wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ;
wndclass.hbrBackground = GetStockObject (WHITE_BRUSH) ;
wndclass.lpszMenuName = NULL ;
wndclass.lpszClassName = szAppName ;
if (!RegisterClass (&wndclass))
{
MessageBox ( NULL, TEXT ("This program requires Windows NT!"),
szAppName, MB_ICONERROR) ;
return 0 ;
}
hwnd = CreateWindow (szAppName, TEXT ("Enhanced MetaFile Demo #1"),
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hInstance, NULL) ;
ShowWindow (hwnd, nCmdShow) ;
UpdateWindow (hwnd) ;
while (GetMessage (&msg, NULL, 0, 0))
{
TranslateMessage (&msg) ;
DispatchMessage (&msg) ;
}
return msg.wParam ;
}
LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam,LPARAM lParam)
{
static HENHMetaFile hemf ;
HDC hdc, hdcEMF ;
PAINTSTRUCT ps ;
RECT rect ;
switch (message)
{
case WM_CREATE:
hdcEMF = CreateEnhMetaFile (NULL, NULL, NULL, NULL) ;
Rectangle (hdcEMF, 100, 100, 200, 200) ;
MoveToEx (hdcEMF, 100, 100, NULL) ;
LineTo (hdcEMF, 200, 200) ;
MoveToEx (hdcEMF, 200, 100, NULL) ;
LineTo (hdcEMF, 100, 200) ;
hemf = CloseEnhMetaFile (hdcEMF) ;
return 0 ;
case WM_PAINT:
hdc = BeginPaint (hwnd, &ps) ;
GetClientRect (hwnd, &rect) ;
rect.left = rect.right / 4 ;
rect.right = 3 * rect.right / 4 ;
rect.top = rect.bottom / 4 ;
rect.bottom = 3 * rect.bottom / 4 ;
PlayEnhMetaFile (hdc, hemf, &rect) ;
EndPaint (hwnd, &ps) ;
return 0 ;
case WM_DESTROY:
DeleteEnhMetaFile (hemf) ;
PostQuitMessage (0) ;
return 0 ;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;
}
在EMF1的窗口消息处理程序处理WM_CREATE消息处理期间,程序首先通过呼叫CreateEnhMetaFile来建立增强型MetaFile。该函数有4个参数,但可以把它们都设为NULL。稍候我将说明这4个参数在非NULL情况下的使用方法。
和CreateMetaFile一样,CreateEnhMetaFile函数传回特定的设备内容句柄。该程序利用这个句柄绘制一个矩形和该矩形的两条对角线。这些函数呼叫及其参数被转换为二进制元的形式并储存在MetaFile中。
最后通过对CloseEnhMetaFile函数的呼叫结束了增强型MetaFile的建立并传回指向它的句柄。该文件句柄储存在HENHMetaFile型态的静态变量中。
在WM_PAINT消息处理期间,EMF1以RECT结构取得程序的显示区域窗口大小。通过调整结构中的4个字段,使该矩形的长和宽为显示区域窗口长和宽的一半并位于窗口的中央。然后EMF1呼叫PlayEnhMetaFile,该函数的第一个参数是窗口的设备内容句柄,第二个参数是该增强型MetaFile的句柄,第三个参数是指向RECT结构的指针。
在MetaFile的建立程序中,GDI得出整个MetaFile图像的尺寸。在本例中,图像的长和宽均为100个单位。在MetaFile的显示程序中,GDI将图像拉伸以适应PlayEnhMetaFile函数指定的矩形大小。EMF1在Windows下执行的三个执行实体如图18-2所示。
图18-2 EMF1得屏幕显示
最后,在WM_DESTROY消息处理期间,EMF1呼叫DeleteEnhMetaFile删除MetaFile。
让我们总结一下从EMF1程序学到的一些东西。
首先,该程序在建立增强型MetaFile时,画矩形和直线的函数所使用的坐标并不是实际意义上的坐标。您可以将它们同时加倍或都减去某个常数,而其结果不会改变。这些坐标只是在定义图像时说明彼此间的对应关系。
其次,为了适于在传递给PlayEnhMetaFile函数的矩形中显示,图像大小会被缩放。因此,如图18-2所示,图像可能会变形。尽管MetaFile坐标指出该图像是正方形的,但一般情况下我们却得不到这样的图像。而在某些时候,这又正是我们想要得到的图像。例如,将图像嵌入一段文书处理格式的文字中时,可能会要求使用者为图像指定矩形,并且确保整个图像恰好位于矩形中而不浪费空间。这样,使用者可通过适当调整矩形的大小来得到正确的纵横比。
然而有时候,您也许希望保留图像最初的纵横比,因为这一点对于表现视觉信息尤为重要。例如,警察的嫌疑犯草图既不能比原型胖也不能比原型瘦。或者您希望保留原来图像的度量尺寸,图像必须是两英…几撸
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -