在进行绘图时,我们经常要面对速度和效率两者之间矛盾。在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。