第八章 用QImage高质量绘图
来源:互联网 发布:91熊猫看书软件 编辑:程序博客网 时间:2024/05/18 00:30
在进行绘图时,我们经常要面对速度和效率两者之间矛盾。在X11和MacOsX系统上,在QWidget和QPixmap绘图要依赖平台自身的绘图引擎。在X11上,与X server的通信很少,Qt只是发送绘图命令而不是真正的绘图数据。这种画法的不足是Qt要收到平台自身绘图引擎的限制。
在X11上,消除锯齿和支持分数坐标这些功能只有在Xserver上安装了XRender扩展才能实现;
在MacOsX平台,它自己的绘图引擎在绘制多段线时使用了和X11和Windows不同的算法,因此得到的结果会有稍许差别。
当准确性比效率重要时,我们可以先绘制在QImage上,然后把结果拷贝到屏幕。在QImage绘图使用Qt自己的绘图引擎,因此在所有平台上都能得到一致的结果。使用这个方法的额外工作是用QImage::Format_RGB32或者QImage::Format_ARGB32_Premutiplied参数创建QImage对象。
QImage::Format_ARGB32_Premutiplied和传统的ARGB32格式(0xaarrggbb)格式完全一致,不同在于红,绿,蓝三个通道值都“乘以”了alpha通道值。这样,0x00到0xFF的RGB颜色值范围变为0x00到alpha通道值。例如50%透明度的蓝色用ARGB格式表示为0x7F0000FF,在用Format_ARGB32_Premutiplied表示时为0x7F00007F,同理,75%透明度的黑绿色在ARGB格式中表示为0x3F008000,在Format_ARGB32_Premutiplied格式中表示为0x3F002000。
如果我们想用消除锯齿的方式绘制一个控件,并希望在没有XRender扩展的X11平台上也得到很好的结果,在原来需要依靠XRender的paintEvent()函数代码如下:
void MyWidget::paintEvent(QPaintEvent *event){ QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing, true); draw(&painter);}下面的代码为重写的paintEvent(),使用了Qt的平台独立的绘图引擎:void MyWidget::paintEvent(QPaintEvent *event){ QImage image(size(), QImage::Format_ARGB32_Premultiplied); QPainter imagePainter(&image); imagePainter.initFrom(this); imagePainter.setRenderHint(QPainter::Antialiasing, true); imagePainter.eraseRect(rect()); draw(&imagePainter); imagePainter.end(); QPainter widgetPainter(this); widgetPainter.drawImage(0, 0, image);}
在上面的代码中,我们创建了一个Format_ARGB32_Premultiplied 格式的QIamge,大小和控件相同,创建一个QPainter绘制这个图像。QPainter::initFrom()调用用控件的设置初始化画笔,刷子和字体。然后想以前一样绘制。最后,创建控件的QPainter对象,把图片拷贝到控件上。
这种方式能够在不同的平台上得到效果一样的结果,但是绘制的文字除外,因为这取决于安装的字体。
Qt绘图引擎的另外一个尤为有用的功能是它能支持混和模式。在进行绘图时,原图像和目标图像能够组合起来。这个混和支持多种绘图操作:如笔,刷子,渐变色,图像等。
缺省的组合模式为QImage::CompositionMode_SourceOver ,即原象素(正在绘制的象素)和目标象素(已经存在的象素)混和,原象素的alpha分量定义为最终的透明度。图8.11显示了不同的模式下绘制的半透明蝴蝶的效果。
Figure 8.11. QPainter’s composition modes
函数QPainter::setCompositionMode()用来设置复合模式。下面的代码创建一个QImage,包含一个蝴蝶和一个棋盘格子的混和:
QImage resultImage = checkerPatternImage;
QPainter painter(&resultImage);
painter.setCompositionMode(QPainter::CompositionMode_Xor);
painter.drawImage(0, 0, butterflyImage);
一个需要注意的问题是,QImage::CompositionMode_Xor模式对alpha通道也同样适用。如果用白颜色0xFFFFFFFF同它自己混和,得到的将是透明色0x00000000,而不是0xFF000000。
- 第八章 用QImage高质量绘图
- 8-3 使用QImage进行高质量绘制(High-Quality Rendering with QImage)
- 使用QImage进行高质量绘制(High-Quality Rendering with QImage)
- 《高质量程序设计(第二版)》第八章中关于指针的问题
- 第八章 高质量C编程规范C++函数的高级特性
- 第八章 MMI 开始绘图
- 第八章:绘图和可视化
- 绘图设备 QPixmap、QBitmap QImage
- 绘图设备 QPixmap、QBitmap QImage
- 绘图设备 QPixmap、QBitmap QImage .
- Qt图片绘图类QPixmap/QImage/QPicture
- QImage
- QImage
- 第七章--高质量的子程序
- (高质量代码专栏)第二章
- (高质量代码专栏)第二章-2
- 用Rails创建高质量Web应用
- 高质量网站关键词用优化吗?
- 【NanoPi Duo试用体验】+matrix_gpio移植
- 自己用VB编写的定时关机软件代码,欢迎大家交流
- 爬坑日记--------搭建springcloud微服务pom
- Linux命令行的艺术(1)-大纲
- RESTful API 设计指南
- 第八章 用QImage高质量绘图
- JspContext源码翻译
- 以旁观者的心态看待
- 堆排序
- WinSCP和PuTTY的搭配与使用--转载
- Kubernetes使用集群联邦实现多集群管理
- 基于Python的flask的开发实战
- 头结点是否为空的问题
- 优秀程序员的书单