大纲:
    基本变换
        介绍和常用API
    窗口-视口转换
        窗口
        视口
        让窗口和视口维持相同宽高比来防止变形
 
基本变换
默认的,QPainter在相关设备的坐标系统上进行操作,但是它也完全支持仿射(affine)坐标变换(仿射变换的具体概念请查看其他资料)。
 
绘图时可以使用常用API:
QPainter::scale()函数缩放坐标系统;
QPainter::rotate()函数顺时针旋转坐标系统;
QPainter::translate()函数平移坐标系统;
QPainter::shear()围绕原点来扭曲坐标系统;
QPainter::save()用来保存QPainter的变换矩阵;
QPainter::restore()用来恢复QPainter的变换矩阵;
 
例如:
    //平移坐标系统,也就是说以(200,150)作为新的原点(0,0)
    QPainter::translate(200, 150); //水平移到200, 垂直移动150
     
    //旋转坐标系统
    //rotate()函数会以原点为中心进行旋转,参数为旋转的角度,正数为顺时针旋转,负数为逆时针旋转
    QPainter::rotate(90);
 
窗口-视口转换
使用QPainter进行绘制时,会使用逻辑坐标进行绘制,然后再转换为绘图设备的物理坐标。
逻辑坐标到物理坐标的映射主要由以下的函数来处理:
QPainter的worldTransform()函数;
QPainter的viewport()函数;
QPainter的window()函数
其中,视口(viewport)表示物理坐标下指定的一个任意矩形,而窗口(window,与以前讲的窗口部件的概念不同)表示逻辑坐标下的相同矩形。默认的,逻辑坐标和物理坐标是重合的,它们都相当于绘图设备上的矩形。
 
1. 窗口(逻辑区域)
窗口代表要逻辑处理的区域,它始终以视口(物理)坐标为最终目标进行映射,他的大小和逻辑位置可以通过QPainter::setWindow()设置,但是无论大小和逻辑位置设置为什么数值,他始终代表着整个视口。
 
例如 有一个实际大小为 200×200 像素的窗口,那么原始状态之下窗口大小也是 200×200,视口大小也是 200×200。
那么在(0,0)位置画一个大小为 100×100 的矩形的时候,它会占视口左上角的4分之一。
painter.drawRect(0,0,100,100);
 
如果这时候我们通过 QPainter::setWindow 修改了窗口位置和大小,例如 setWindow(-50,-50,100,100)
设置完后,窗口代表的还是整个视口(物理),但是映射的数值有所不同:
此时窗口的逻辑坐标(-50,-50)对应视口(物理)坐标的(0,0)。
而窗口的逻辑大小成为了100×100的单位长度,也就是用100个单位长度代表原本物理大小的200像素,所以每一个单位长度就是实际的2像素。
 
因为QPainter是以窗口(逻辑)坐标为基础的,所以这时候画一个位置为(-50,-50),大小为 50,50的矩形。
painter.drawRect(-50,-50,50,50);还是和以前画的是一样的的效果。
 
也就是说,我们将窗口(逻辑)宽 100px 映射成视口(物理)宽 200px,窗口高 100px 映射成物理高 200px,逻辑-50,-50映射成物理的0,0。
 
 
2. 视口(物理区域)
现在来改变视口的属性,先用painter.setViewPort(0,0,100,100);
上面语句的作用就是把把视口的坐标设置为绘图区的左上角(0,0)位置,大小设置为绘图区的一半,因为绘图区是(200×200),而我们把视口设置为(100×100)。
 
即现在实际的绘图区为绘图设备的左上角的4分之一。
也就是说,我们将视口(物理)宽 100px 映射成窗口(逻辑)宽 200px,视口高 100px 映射成窗口高 200px。
这时候我们再用
painter.drawRect(0,0,100,100);
画一个矩形,实际显示是怎么样的呢?看下图:
绘制出来的是dialog的16分之一了,为什么会这样呢?
 
前面我们讲过窗口坐标始终以视口(逻辑)坐标为最终目标进行映射,而原来没有经过修改的窗口的属性为以左上角为原点,大小为200×200单位长度,我们修改视口大小为100×100像素后,窗口的200单位长度就映射到100像素的视口长度上,即每一单位长度为0.5像素,所以绘制出来的结果就是100×0.5=50像素,所以长和高都是dialog的4分之一,面积就是16分之一了。
 
3. 让视口和窗口维持相同的宽高比来防止变形
int side = qMin(width(), height()); //取长度和宽度的最小值
int x = (width() - side/2);
int y = (height() - side/2);
painter.setViewport(x,y,side,side); //物理的(x,y)映射成逻辑的(0,0);物理的宽side,映射成逻辑的width()......
 
参考:
 
本篇源代码
 
 

[Qt2D绘图]-03坐标系统之坐标变换的更多相关文章

  1. [Qt2D绘图]-02坐标系统&&抗锯齿渲染

    本节的内容可以在帮助中通过Coordinate System关键字查看. 或者入门可以看<Qt Creator 快速入门>这本书.强烈推荐入门使用.下面的内容为本书的阅读笔记,喜欢的可以买 ...

  2. 【转 | 侵删】2D 绘图技术中的坐标系统与坐标变换

    本文介绍在 2D 绘图技术中的坐标系统和坐标变换的相关知识.同时介绍 Kity 在这方面提供的 API .希望这些知识对于需要进行图形应用开发的同学会有所帮助. 锤子的故事 很久以前,有一个画家,他很 ...

  3. [Qt2D绘图]-06QPainter的复合模式&&双缓冲绘图&&绘图中的其他问题

    本篇读书笔记主要记录QPainter的复合模式&&双缓冲绘图&&绘图中的其他问题   大纲:     复合模式     双缓冲绘图     绘图中的其他问题       ...

  4. [Qt2D绘图]-04绘制文字&&绘制路径

    注:学习自<Qt Creator 快速入门>第三版.   文档中的示例参考 Qt Example推荐:Painter Paths Example和Vector Deformation   ...

  5. Delphi下OpenGL2d绘图(03)-画线

    一.前言 画线与画点基本上代码是相同.区别在于glBegin()的参数.绘制的框架代码可以使用 Delphi下OpenGL2d绘图(01)-初始化 中的代码.修改的部份为 Draw 函数的内容. 二. ...

  6. [Qt2D绘图]-05绘图设备-QPixmap&&QBitmap&&QImage&&QPicture

    这篇笔记记录的是QPainterDevice(绘图设备,可以理解为一个画板) 大纲:     绘图设备相关的类:QPixmap QBitmap QImage QPicture     QPixmap ...

  7. [Qt2D绘图]-01Qt2D绘图基本绘制和填充

    Qt的文档位置为 Paint System   大纲:     简介     先看一个小例子     基本的绘制和填充         使用画笔(Qpen)         使用画刷(QBrush) ...

  8. qt 坐标变换

    原文链接:Qt学习之路(28): 坐标变换   同很多坐标系统一样,QPainter的默认坐标的原点(0, 0)位于屏幕的左上角,X轴正方向是水平向右,Y轴正方向是竖直向下.在这个坐标系统中,每个像素 ...

  9. Qt学习之路(28): 坐标变换

    经过前面的章节,我们已经能够画出一些东西来,主要就是使用QPainter的相关函数.今天,我们要看的是QPainter的坐标系统.   同很多坐标系统一样,QPainter的默认坐标的原点(0, 0) ...

随机推荐

  1. 这一次搞懂SpringMVC原理

    @ 目录 前言 正文 请求入口 组件初始化 调用Controller 参数.返回值解析 总结 前言 前面几篇文章,学习了Spring IOC.Bean实例化过程.AOP.事务的源码和设计思想,了解了S ...

  2. 版本控制工具 GIT入门教程

    GIT 在团队中的中作流程 1.每个程序员在自己的分支上进行开发 2.主程序猿/Leader合并程序员程序 3.程序员之间也可以对一下提交冲突进行合并 下载和安装 GIT官方网址:http:// gi ...

  3. ElasticSearch中的sort排序和filedData作用

    默认情况下,ElasticSearch 会根据算分进行排序: 可以使用 sort API 指定排序的规则: POST /kibana_sample_data_ecommerce/_search { & ...

  4. loadRunnner中90%的响应时间

    参考博客https://blog.csdn.net/lengyue_112/article/details/1095320?utm_source=blogxgwz4 LR在场景执行完了会出个报告,其中 ...

  5. C++核心内容和机制

    备注:不局限与C++版本   一. 基础知识 数据类型和POD/Trivial 数据类型: 类型转换: NULL和nullptr: 操作符重载: 全局静态变量和成员静态变量的申明和初始化: 左值和右值 ...

  6. Python三大器之迭代器

    Python三大器之迭代器 迭代器协议 迭代器协议规定:对象内部必须提供一个__next__方法,对其执行该方法要么返回迭代器中的下一项(可以暂时理解为下一个元素),要么就引起一个Stopiterat ...

  7. python高级-闭包-装饰器

    闭包内容: 匿名函数:能够完成简单的功能,传递这个函数的引用,只有功能 普通函数:能够完成复杂的功能,传递这个函数的引用,只有功能 闭包:能够完成较为复杂的功能,传递这个闭包中的函数以及数据,因此传递 ...

  8. P2136 拉近距离

    我也想有这样的爱情故事,可惜我单身 其实这道题就是一个比较裸的最短路问题.对于一个三元组 (S,W,T) ,S其实就是一个端点,而W就是到达的端点,连接两个端点的边长为-T,注意要取一个相反数,这样才 ...

  9. Python内置函数和内置常量

    Python内置函数 1.abs(x) 返回一个数的绝对值.实参可以是整数或浮点数.如果实参是一个复数,返回它的模. 2.all(iterable) 如果 iterable 的所有元素为真(或迭代器为 ...

  10. 修改CentOS7登录欢迎界面信息

    vi /etc/issue 添加自己喜欢的内容,保存即可. 特殊字符的含义: \d 本地端时间的日期: \l 显示第几个终端机接口: \m 显示硬件的等级 (i386/i486/i586/i686.. ...