what size
SetMapMode函数
SetWindowExtEx函数
SetViewportExtEx函数
MM_ANISOTROPIC映射模式
MM_ISOTROPIC映射模式
MM_HIENGLISH映射模式
MM_HIMETRIC映射模式
MM_LOENGLISH映射模式
MM_LOMETRIC映射模式
MM_TEXT映射模式
MM_TWIPS映射模式
/*---------------------------------------------------------
s027.c---a daily practice
027.c WIN32 API 每日一练
第27个例子:GDI映射模式
SetMapMode函数
SetWindowExtEx
SetViewportExtEx
MM_ANISOTROPIC映射模式
MM_ISOTROPIC
MM_HIENGLISH
MM_HIMETRIC
MM_LOENGLISH
MM_LOMETRIC
MM_TEXT
MM_TWIPS
By zhyjhacker 2023.06.29
-----------------------------------------------------------*/
#include<Windows.h>
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
int WINAPI WinMain(HINSTANCE hinstance, HINSTANCE hPrevInstance,
PSTR szCmdLine, int iCmdShow)
{
static TCHAR szAppName[] = TEXT("WhatSize.c");//移植你更改的地方
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 = (HBRUSH)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("Device Capabilities"),//更新
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hinstance, NULL);
ShowWindow(hwnd, iCmdShow);
UpdateWindow(hwnd);
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return msg.wParam;
}
//显示装口客户区的尺寸
void show(HWND hwnd, HDC hdc, int xText, int yText, int iMapMode, TCHAR* szMapMode)
{
TCHAR szBuffer[60];
RECT rect;
SaveDC(hdc);//保存当前设备环境句柄
SetMapMode(hdc, iMapMode);//指定新的映射模式
GetClientRect(hwnd, &rect);//获取窗口客户区的大小---使用当前HDC中映射模式的度量单位
DPtoLP(hdc, (PPOINT)&rect, 2);//设备坐标转换成逻辑坐标
RestoreDC(hdc, -1);//恢复原来设备环境,-1恢复最近保存的环境
TextOut(hdc, xText, yText, szBuffer,
wsprintf(szBuffer, TEXT("%-20s %7d %7d %7d %7d"), szMapMode,
rect.left, rect.right, rect.top, rect.bottom));
}
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
static TCHAR szHeading[] = TEXT("mapping Mode Left Right Top bottom");
static TCHAR szUndLine[] = TEXT("------------ ---- ----- --- ------");
static int cxChar, cyChar;//字符的宽度和高度
HDC hdc;
PAINTSTRUCT ps;
TEXTMETRIC tm;
switch (message)
{
case WM_CREATE:
hdc = GetDC(hwnd);
//选入字体
SelectObject(hdc, GetStockObject(SYSTEM_FIXED_FONT));//等宽字体
//获取字体参数
GetTextMetrics(hdc, &tm);
cxChar = tm.tmAveCharWidth;//平均字符宽度
cyChar = tm.tmHeight + tm.tmExternalLeading;//总高度
ReleaseDC(hwnd, hdc);
return 0;
case WM_PAINT:
hdc = BeginPaint(hwnd, &ps);
//选入字体
SelectObject(hdc, GetStockObject(SYSTEM_FIXED_FONT));//选择备用笔,画
//指定新的映射模式,逻辑单位被映射到具有任何方所轴的任意单位
SetMapMode(hdc, MM_ANISOTROPIC);//拉伸图形以满足需要
//SetMapMode(hdc, MM_TEXT);
//SetMapMode(hdc, MM_ANISOTROPIC)//
//SetWindowExtEx和SetViewportExtEx决定了窗口和视口之间的缩放因子
//只有 MM_ANISOTROPIC 和 MM_ISOTROPIC 两种映射模式需要设置窗口和视口设备范围
//设置窗口设备上下文的水平和垂直范围,改窗口是指一面空间的逻辑坐标系
SetWindowExtEx(hdc, 1, 1, NULL);//x轴=1像素,y轴=1像素
//设置视窗上下问的水平和垂直范围
SetViewportExtEx(hdc, cxChar, cyChar, NULL);//x轴=1个字符宽,y轴=1个字符宽
//该映射模式显示范围不受限制,可以任何拉伸以满足需求
TextOut(hdc, 1, 1, szHeading, lstrlen(szHeading));
TextOut(hdc, 1, 2, szUndLine, lstrlen(szUndLine));
//在距离左边1个字符,距离顶部3个字节出的客户区显示文本
//每个逻辑单元映射到一个设备像素
show(hwnd, hdc, 1, 3, MM_TEXT, TEXT("TEXT(pixels)"));
//每个逻辑单元映射到0.1毫米
show(hwnd, hdc, 1, 4, MM_LOMETRIC, TEXT("LOMETRIC(.1mm)"));
//每个逻辑单元映射到0.01毫米
show(hwnd, hdc, 1, 5, MM_HIMETRIC, TEXT("HIMETRIC(.01 mm)"));
//每个逻辑单元映射到0.1英寸
show(hwnd, hdc, 1, 6, MM_LOENGLISH, TEXT("LOENGLISH(0.1 in)"));
//每个逻辑单元映射到.01英寸
show(hwnd, hdc, 1, 7, MM_HIENGLISH, TEXT("HIENGLISH(.001 in)"));
//逻辑是,每个逻辑单位都映射到打印机点的二十分之一
show(hwnd, hdc, 1, 8, MM_TWIPS, TEXT("TWIPS(1/1440 in)"));
EndPaint(hwnd, &ps);
return 0;
case WM_DESTROY:
PostQuitMessage(0);
return 0;
}
return DefWindowProc(hwnd, message, wParam, lParam);
}