概述

调用 SetMapMode 函数可以设置映射模式:

int SetMapMode(
HDC hdc, // 设备环境句柄
int fnMapMode // 要设置的映射模式
);

同样,调用 GetMapMode 函数可以获取映射模式:

int GetMapMode(
HDC hdc // 设备环境句柄
);

注:

  • 默认情况下,映射模式是 MM_TEXT,以像素为单位操作。

  • 映射模式的逻辑坐标只对以设备环境句柄为参数的 GDI 函数有效,非 GDI 函数将继续使用设备坐标

设备坐标系统

  1. 屏幕坐标系统:屏幕左上角坐标为(0,0)

  2. 全窗口坐标系统:窗口边框左上角坐标为(0,0)
  3. 设备坐标系统:客户区左上角坐标为(0,0)

设备坐标和逻辑坐标

设备坐标指视口坐标,逻辑坐标指窗口坐标

在 MM_TEXT 映射模式下,逻辑坐标与设备坐标重合,更改映射模式后,逻辑坐标对设备坐标的映射方法将发生改变。

窗口和视口

调用 SetViewPortOrgEx 函数可以设置视口(设备坐标)的原点

调用 SetWindowOrgEx 函数可以设置窗口(逻辑坐标)的原点

(实际上,这两个函数是改变了设备坐标到逻辑坐标的映射方式,而设备坐标永远是(0,0))

例如 SetViewPortOrgEx ( hdc, 40, 40 ) 其实是将设备坐标(40,40)映射到了逻辑坐标原点(0,0)

例如 SetWindowOrgEx ( hdc, 40, 40 ) 其实是将逻辑坐标(40,40)映射到了设备坐标原点(0,0)

坐标转换

调用 LPToDP 函数可以将逻辑坐标转换成设备坐标

调用 DPToLP 函数可以将设备坐标转换成逻辑坐标

各同向性和各异向性映射模式

当第一次指定各同向性(MM_ISOTROPIC)和各异向性(MM_ANISOTROPIC)的映射模式时, 和 MM_LOMETRIC 映射模式有相同的效果

调用 SetViewPortExtEx 函数可以设置视口(设备坐标)区域的大小(仅在各同向性(MM_ISOTROPIC)和各异向性(MM_ANISOTROPIC)的映射模式中起作用)

调用 SetWindowExtEx 函数可以设置窗口(逻辑坐标)区域的大小(仅在各同向性(MM_ISOTROPIC)和各异向性(MM_ANISOTROPIC)的映射模式中起作用)

(SetWindowExtEx 和 SetViewPortExtEx 函数,必须先后都调用,SetWindowExtEx 最好在 SetViewPortExtEx 前调用,这两个函数的本质是设置一种从逻辑坐标到设备坐标的缩放比例,实际上最终要转换到用以下两个公式来换算逻辑坐标位置映射到的设备坐标位置)

MM_ISOTROPIC 和 MM_ANISOTROPIC 的区别:

MM_ISOTROPIC 会将横纵坐标的缩放比例保持一致(以比例较小的为标准),故当逻辑区域大小改变的时候,图形不会发生拉伸现象,会保持横纵缩放比例一致。

MM_ANISOTROPIC 不会将横纵坐标缩放比例保持一致,故当逻辑区域大小改变的时候,图形将可能会发生拉伸现象。

WHATSIZE 示例程序

#include <windows.h>
#include <strsafe.h> void Show(HWND hwnd, HDC hdc, int xText, int yText, int iMapMode, LPCTSTR szMapMode) {
TCHAR szBuffer[60];
size_t ccLength;
RECT rcClient; SaveDC(hdc);
SetMapMode(hdc, iMapMode);
GetClientRect(hwnd, &rcClient);
DPtoLP(hdc, (LPPOINT)&rcClient, 2);
RestoreDC(hdc, -1); StringCchPrintf(szBuffer, sizeof(szBuffer) / sizeof(TCHAR), TEXT("%-20s %7d %7d %7d %7d"), szMapMode, rcClient.left, rcClient.right, rcClient.top, rcClient.bottom);
StringCchLength(szBuffer, sizeof(szBuffer) / sizeof(TCHAR), &ccLength);
TextOut(hdc, xText, yText, szBuffer, ccLength);
} LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { HDC hdc;
static int cxChar, cyChar;
PAINTSTRUCT ps;
TEXTMETRIC tm;
static TCHAR szHeading[] = TEXT("Mapping Mode Left Right Top Bottom");
static TCHAR szUndLine[] = TEXT("------------ ---- ----- --- ------");
size_t ccLength; 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);
SetWindowExtEx(hdc, 1, 1, NULL);
SetViewportExtEx(hdc, cxChar, cyChar, NULL); StringCchLength(szHeading, sizeof(szHeading) / sizeof(TCHAR), &ccLength);
TextOut(hdc, 1, 1, szHeading, ccLength); StringCchLength(szUndLine, sizeof(szUndLine) / sizeof(TCHAR), &ccLength);
TextOut(hdc, 1, 2, szUndLine, ccLength); Show(hwnd, hdc, 1, 3, MM_TEXT, TEXT("TEXT (pixels)"));
Show(hwnd, hdc, 1, 4, MM_LOMETRIC, TEXT("LOMETRIC (.1 mm)"));
Show(hwnd, hdc, 1, 5, MM_HIMETRIC, TEXT("HIMETRIC (.01 mm)"));
Show(hwnd, hdc, 1, 6, MM_LOENGLISH, TEXT("LOENGLISH (.01 in)"));
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);
} int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { LPCTSTR lpszClassName = TEXT("WHATSIZE");
LPCTSTR lpszWindowName = TEXT("WHATSIZE Program");
WNDCLASS wndclass;
HWND hwnd;
MSG msg; wndclass.cbClsExtra = 0;
wndclass.cbWndExtra = 0;
wndclass.hbrBackground = (HBRUSH) GetStockObject(WHITE_BRUSH);
wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wndclass.hInstance = hInstance;
wndclass.lpfnWndProc = WndProc;
wndclass.lpszClassName = lpszClassName;
wndclass.lpszMenuName = NULL;
wndclass.style = CS_HREDRAW | CS_VREDRAW; if (!RegisterClass(&wndclass)) {
MessageBox(NULL, TEXT("window class registering failed!"), TEXT("Error"), MB_ICONERROR);
return 0;
} hwnd = CreateWindow(
lpszClassName,
lpszWindowName,
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;
}

GDI 映射模式(11)的更多相关文章

  1. Windows GDI 映射模式(出自:Windows程序设计第5版-珍藏版)

    GDI映射模式(mapping mode):和映射模式紧密相关的还有4个其它的设备环境属性:1.窗口原点(window origin)2.视口原点(viewport origin)3.窗口范围(win ...

  2. Windows绘图中的GDI映射模式

    对Windows编程新手来说,GDI编程中最困难的部分就是映射模式(Mapping Mode). 什么是映射模式? 我们知道,GDI画图是在设备描述表这个逻辑意义上的显示平面上进行,其使用的是逻辑坐标 ...

  3. GDI绘图中的映射模式CDC::SetMapMode()

    原文链接:http://blog.csdn.net/charlessimonyi/article/details/8264572 在GDI绘图前,一般要设置映射模式.映射模式是什么呢?它是逻辑长度单位 ...

  4. MFC坐标空间与映射模式

    逻辑坐标:使用GDI绘图时使用的坐标系 设备坐标系:实际设备(显示器.打印机)的坐标系,即我们实际看到的坐标系. 坐标空间 在Windows NT/2000中Win32 API中支持以下四层坐标空间: ...

  5. Visual C++ 打印编程技术-编程基础-映射模式

    映射模式: Visual C++ 中采用的坐标映射方式使得用户图形坐标和输出设别的像素完全一致. eg:当屏幕的像素大小为800X600时,每英寸包含屏幕像素为96,打印机则需要几倍的点数才能达到同样 ...

  6. php模式-数据映射模式

    概念:简言之,数据映射模式就是将对象和数据存储映射起来,对一个对象的操作会映射为对数据存储的操作. 深入理解:数据映射,是在持久化数据存储层(一般是关系型数据库)和驻于内存的数据表现层之间进行双向数据 ...

  7. php设计模式 数据对象映射模式

    数据对象映射模式,是将对象和数据存储映射起来,对一个对象的操作会映射为对数据存储的操作. 在代码中实现数据对象映射模式,实现一个ORM类,将复杂的sql语句映射成对象属性的操作.对象关系映射(Obje ...

  8. Hibernate的关联映射——单向1-1关联

    Hibernate的关联映射--单向1-1关联 对于单向的1-1关联关系,需要在持久化类里增加代表关联实体的成员变量,并为该成员变量添加setter方法和getter方法.从持久化类的代码上看,单向1 ...

  9. PHP 设计模式 笔记与总结(10)数据对象映射模式 2

    [例2]数据对象映射模式结合[工厂模式]和[注册模式]的使用. 入口文件 index.php: <?php define('BASEDIR',__DIR__); //定义根目录常量 includ ...

随机推荐

  1. no_merge hint

    This is tested in 10gR2. SQL> select * from v$version; BANNER ----------------------------------- ...

  2. Navicat 提示Cannot create oci environment 解决方式

    一直在使用Navicat,这是一个数据库client软件.能连接多种不同类型的数据库,给我们的日常的工作带来了不少的便捷.近期.我在电脑上安装了orcale,然后,Navicat就莫名其妙的不能连接o ...

  3. jenkins配置邮箱遇到的问题

    错误一:发送测试邮件测试配置没有填写接收者的邮箱 原因:没有写接收者的邮箱 2.写了接受者的邮箱 密码错误 解决办法:qq邮箱>设置>账户,发送短信后点我已发送,就会接收到密码 3.发送时 ...

  4. 【POJ 1201】 Intervals(差分约束系统)

    [POJ 1201] Intervals(差分约束系统) 11 1716的升级版 把原本固定的边权改为不固定. Intervals Time Limit: 2000MS   Memory Limit: ...

  5. 【NOIP2018】为什么这么无力啊

    菜鸡又要爆零了 辛辛苦苦背板子结果考时候脑子一片空白 第一题线段树调了半小时 看完三道题两道写暴搜一道写暴力(说是暴搜,觉得更像写了个背包) 别提暴搜还忘记剪枝. . . . . . 我觉得考场上最菜 ...

  6. C#获取本机Sql Serverserver名

    private void Form2_Load(object sender, EventArgs e) { listBox1.Items.Clear(); SQLDMO.Application SQL ...

  7. Java学习笔记之从C++转Java

    之前一直是做C++后台开发的,习惯了命令行和g++,由于工作原因现在开始转java. 1.参考书籍:java编程思想(think in java),java核心技术(core java); 2.怎么在 ...

  8. 遍历WPF DataGrid单元格

    using System.Windows.Controls; using System.Windows.Controls.Primitives; using System.Windows.Media; ...

  9. position中的absolute、fixed区别

    absolute: 绝对定位,相对于body.   fixed: 固定定位,相对于浏览器视窗,不随滚动条的滚动而滚动. 这两个属性概念比较模糊,一般在做左边列表菜单,右边内容区域的时候会用到这样的定位 ...

  10. 生成器模式(Builder)C++实现

    意图:将一个复杂对象的创建与它的表示分离,使得同样的构建过程可以创建不同的表示. 适用性:1.当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式时. 2.当构建过程必须允许被构建的对象有 ...