Qt组件中的双缓冲无闪烁绘图

双缓冲绘图
在Qt4中,所有的窗口部件默认都使用双缓冲进行绘图。使用双缓冲,可以减轻绘制的闪烁感。在有些情况下,用户要关闭双缓冲,自己管理绘图。下面的语句设置了窗口部件的Qt::WA_PaintOnScreen属性 ,就关闭了窗口部件的双缓冲.
mywidget->setAttribute(Qt::WA_PaintOnScreen);

由于Qt4不再提供异或笔,组合模式QPainter::CompostionMode_Xor()并不是异或笔,Qt4只提供了QRubberBand实现矩形和直线的绘图反馈。因此要实现在绘图中动态
反馈必须使用其他方法。程序中使用双环冲来解决这个问题。在绘图过程中,一个缓冲区绘制临时内存,一个缓冲区保存绘制好的内容,最后进行合并。
在交互绘图过程中,程序将图像缓冲区复制到临时缓冲区,并在临时缓冲区上绘制,绘制完毕在将结果复制到图像缓冲区,如果没有交互复制,则直接将图像缓冲区绘制显示到屏幕上。

Qt组件中的双缓冲无闪烁绘图
闪烁首先,要想把闪烁减弱,请设置组件的背景模式为NoBackground. 
setBackgroundMode(NoBackground);

其次,重载组件的paintEvent()函数,如下改写: 
void MyWidget::paintEvent(QPaintEvent *e) 

QRect ur=e->rect();//得到组件尺寸 
QPixmap pix(ur.size());//以此为参数创建一个位图变量 
pix.fill(this,ur.topLeft());//填充位图 
QPainter p(&pic);//以位图为参数创建一个QPainter 对象

p.translate(-ur.x(),-ur.y());//在QPainter 上绘画 
//......//Drawing

p.End();//绘画完毕

bitBlt(this,ur.topLeft().&pix);//把位图贴到组件上

//注从qt4开始,bitBlt函数不在使用,取而代之的是drawPixmap。
}

// 这是能随机绘点的关键,没有设置此属性,默认相当于每次Qt都会完整的将上一次的屏幕擦除,

// 新版的Qt中已经没有了repaint(bool)接口了。

w.setAttribute(Qt::WA_OpaquePaintEvent);

老电视机雪花效果中每次都需要擦除重绘避免点的叠加所以一下语句注释掉

//    w.setAttribute(Qt::WA_OpaquePaintEvent);

(

以下是私有函数的实现:

void Plotter::updateRubberBandRegion()

{

QRect rect = rubberBandRect.normalized();

update(rect.left(), rect.top(), rect.width(), 1);

update(rect.left(), rect.top(), 1, rect.height());

update(rect.left(), rect.bottom(), rect.width(), 1);

update(rect.right(), rect.top(), 1, rect.height());

}

函数updateRubberBand()在mousePressEvent(),mouseMoveEvent()和mouseReleaseEvent()中被调用,用来删除或者重新绘制橡皮线。函数中调用了四次update(),用四个绘制事件完成由橡皮线(两条垂直和水平的线)组成的四个小矩形的绘制。Qt也提供了一个类QRubberBand用来绘制橡皮线,但是控件自己提供的绘制函数会更好

void Plotter::refreshPixmap()

{

pixmap = QPixmap(size());

pixmap.fill(this, 0, 0);

QPainter painter(&pixmap);

painter.initFrom(this);

drawGrid(&painter);

drawCurves(&painter);

update();

}

函数refreshPixmap()把plot绘制到图片上,并且更新显示。首先我们把图片的大小调整为和当前控件大小相同,然后用控件的背景颜色填充整个图片。这个颜色是当前调色版的“dark”部分,因为在Plotter构造函数中调用setBackgroundRole() 。如果背景用的刷子是非实心的(solid brush,刷子的样式,只有颜色,没有花纹的那种最简单的),QPixmap::fill()需要知道控件中刷子的偏移量,以便图片对齐刷子模式。这里图片对应整个控件,因此偏移位置为(0,0)。

接下来我们创建了一个QPainter对象来绘制图片,QPainter::initFrom()设置绘制图片所需画笔,背景和字体,参数this表示这些设置和Plotter控件的相应设置是一致的。然后我们调用drawGrid(),drawCurves()绘制网格和曲线。最后,update()函数安排整个控件的绘制事件,在painteEvent()函数中把图片拷贝到控件上。

void Plotter::drawGrid(QPainter *painter)

{

QRect rect(Margin, Margin,

width() - 2 * Margin, height() - 2 * Margin);

if (!rect.isValid())

return;

PlotSettings settings = zoomStack[curZoom];

QPen quiteDark = palette().dark().color().light();

QPen light = palette().light().color();

for (int i = 0; i <= settings.numXTicks; ++i) {

int x = rect.left() + (i * (rect.width() - 1)

/ settings.numXTicks);

double label = settings.minX + (i * settings.spanX()

/ settings.numXTicks);

painter->setPen(quiteDark);

painter->drawLine(x, rect.top(), x, rect.bottom());

painter->setPen(light);

painter->drawLine(x, rect.bottom(), x, rect.bottom() + 5);

painter->drawText(x - 50, rect.bottom() + 5, 100, 15,

Qt::AlignHCenter | Qt::AlignTop,

QString::number(label));

}

for (int j = 0; j <= settings.numYTicks; ++j) {

int y = rect.bottom() - (j * (rect.height() - 1)

/ settings.numYTicks);

double label = settings.minY + (j * settings.spanY()

/ settings.numYTicks);

painter->setPen(quiteDark);

painter->drawLine(rect.left(), y, rect.right(), y);

painter->setPen(light);

painter->drawLine(rect.left() - 5, y, rect.left(), y);

painter->drawText(rect.left() - Margin, y - 10, Margin - 5, 20,

Qt::AlignRight | Qt::AlignVCenter,

QString::number(label));

}

painter->drawRect(rect.adjusted(0, 0, -1, -1));

}

函数drawGrid()在坐标轴和曲线的下面绘制网格。这个区域由一个矩形确定,如果控件太小,则不绘制立即返回。第一个循环绘制网格的垂直线,及沿x坐标轴的刻度。第二个循环绘制网格的水平线,及沿y坐标轴的刻度。最后,沿边界绘制一个矩形。drawText()绘制数字,对应两个坐标轴上刻度的标记。

函数painter->drawText()语法如下:

painter->drawText(x, y, width, height, alignment, text);

其中(x,y,width,height)所确定的矩形,alignment确定文字在矩形中的位置。

void Plotter::drawCurves(QPainter *painter)

{

static const QColor colorForIds[6] = {

Qt::red, Qt::green, Qt::blue, Qt::cyan, Qt::magenta, Qt::yellow

};

PlotSettings settings = zoomStack[curZoom];

QRect rect(Margin, Margin,

width() - 2 * Margin, height() - 2 * Margin);

if (!rect.isValid())

return;

painter->setClipRect(rect.adjusted(+1, +1, -1, -1));

QMapIterator<int, QVector<QPointF> > i(curveMap);

while (i.hasNext()) {

i.next();

int id = i.key();

const QVector<QPointF> &data = i.value();

QPolygonF polyline(data.count());

for (int j = 0; j < data.count(); ++j) {

double dx = data[j].x() - settings.minX;

double dy = data[j].y() - settings.minY;

double x = rect.left() + (dx * (rect.width() - 1)

/ settings.spanX());

double y = rect.bottom() - (dy * (rect.height() - 1)

/ settings.spanY());

polyline[j] = QPointF(x, y);

}

painter->setPen(colorForIds[uint(id) % 6]);

painter->drawPolyline(polyline);

}

}

函数drawCurves()在网格的上层绘制曲线。调用了QPainter::setClipRect()函数设置QPainter的剪切区域为包含曲线的矩形区域(不包括四周的间隙和图片的外框)。QPainter会忽略这个区域外的象素。

然后我们使用Java风格的迭代器,遍历所有的曲线,对每一条曲线,遍历它所有的QPointF点。函数key()得到曲线的id,value()函数得到曲线的QVector<QPointF>类型的数据。内层循环把每个QPointF记录的plotter坐标转换为控件坐标,把结果保存在polyline变量中。

转换坐标后,我们设置画笔的颜色(使用函数前面预定义的颜色),调用drawPolyline()绘制曲线,经过所有的曲线上的点。

http://blog.csdn.net/Last_Impression/archive/2008/05/20/2463647.aspx

时间: 2024-08-01 22:46:52

Qt组件中的双缓冲无闪烁绘图的相关文章

OpenGL中实现双缓冲技术

在OpenGL中实现双缓冲技术的一种简单方法: 1.在调用glutInitDisplayMode函数时, 开启GLUT_DOUBLE,即glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE);.这里将我们惯用的GLUT_SINGLE替换为GLUT_DOUBLE,意为要使用双缓冲而非单缓冲. 2. 调用glutDisplayFunc(display)注册回调函数时, 在回调函数中所有绘制操作完成后调用glutSwapBuffers()交换两个缓冲区指针. 3. 调用

c++双缓冲技术绘图避免闪烁

当数据量很大时,绘图可能需要几秒钟甚至更长的时间,而且有时还会出现闪烁现象,为了解决这些问题,可采用双缓冲技术来绘图. 双缓冲即在内存中创建一个与屏幕绘图区域一致的对象,先将图形绘制到内存中的这个对象上,再一次性将这个对象上的图形拷贝到屏幕上,这样能大大加快绘图的速度.双缓冲实现过程如下: 1.在内存中创建与画布一致的缓冲区 2.在缓冲区画图 3.将缓冲区位图拷贝到当前画布上 4.释放内存缓冲区 在图形图象处理编程过程中,双缓冲是一种基本的技术.我们知道,如果窗体在响应WM_PAINT消息的时候

winform的双缓冲

搜搜winform的双缓冲,就会发现网络上有很多文章,乱七八糟说的不明不白.第一种方案: SetStyle(ControlStyles.UserPaint, true); SetStyle(ControlStyles.AllPaintingInWmPaint, true); // 禁止擦除背景. SetStyle(ControlStyles.DoubleBuffer, true); // 双缓冲 第二种方案: this.DouleBuffered=true 第三种方案: Bitmap bimtB

【转】双缓冲(Double Buffer)原理和使用

原文出自:http://blog.csdn.net/xiaohui_hubei/article/details/16319249 一.双缓冲作用 双缓冲甚至是多缓冲,在许多情况下都很有用.一般需要使用双缓冲区的地方都是由于"生产者"和"消费者"供需不一致所造成的.这样的情况在很多地方后可能会发生,使用多缓冲可以很好的解决.我举几个常见的例子: 例 1. 在网络传输过程中数据的接收,有时可能数据来的太快来不及接收导致数据丢失.这是由于"发送者"和&

双缓冲(Double Buffer)原理和使用

一.双缓冲作用 双缓冲甚至是多缓冲,在许多情况下都很有用.一般需要使用双缓冲区的地方都是由于“生产者”和“消费者”供需不一致所造成的.这样的情况在很多地方后可能会发生,使用多缓冲可以很好的解决.我举几个常见的例子: 例 1. 在网络传输过程中数据的接收,有时可能数据来的太快来不及接收导致数据丢失.这是由于“发送者”和“接收者”速度不一致所致,在他们之间安排一个或多个缓冲区来存放来不及接收的数据,让速度较慢的“接收者”可以慢慢地取完数据不至于丢失.  例2. 再如,计算机中的三级缓存结构:外存(硬

OpenGL学习03_双缓冲DoubleBuffering

双缓冲的是什么? 我们看电视时,看到的屏幕称为OSD层,也就是说,只有在OSD层上显示图像我们才能看到.现在,我需要创建一个虚拟的.看不见但是可以在上面画图(比如说画点.线)的OSD层,我称之为offscreen(后台缓冲区).这个offscreen存在于内存中,我们在上面画图,这个offscreen上面的东西可以显示在OSD层上,需要一个创建这个offscreen的函数,返回这个offscreen的句柄(整型指针).宽度.高度.指向新建offscreen数据缓冲区的指针,该缓冲区是一个在函数外

双缓冲技术讲解

笔者介绍:姜雪伟,IT公司技术合伙人,IT高级讲师,CSDN社区专家,特邀编辑,畅销书作者,国家专利发明人;已出版书籍:<手把手教你架构3D游戏引擎>电子工业出版社和<Unity3D实战核心技术详解>电子工业出版社等. CSDN视频网址:http://edu.csdn.net/lecturer/144 首先要搞清楚计算机运行原理,计算机载运行时是将将最大的任务分解成多个任务,然后一个接一个地执行. 一个典型的例子,每个游戏引擎必须解决的问题是渲染. 当游戏画出用户看到的世界时,比如

MFC双缓冲绘图实例

本人之前一直了解双缓冲绘图的基本原理,但是在研究很久之后才大概知道具体的使用过程,本文将详细介绍本人在实际项目中使用双缓冲绘图的案例. 实现功能:主界面显示某张包含人脸的图片,通过dlib detector获取到人脸上的68个关键点,绘制在图片上显示,然后通过鼠标拖动图片上的关键点,调整位置,之后保存.双缓冲主要能够解决拖动关键点时屏幕闪烁的问题,本文主要侧重在双缓冲的实现,其他功能概不介绍. 具体实现: 1.定义全局变量: CDC dc_mem://内存绘制dc CDC *dc://绘图dc

控件的重绘和双缓冲技术

处理重绘和无效操作: 重新定制控件行为和外观 Void Invalidate(); void Invalidate ( Rectangle ); //使控件的特定区域无效并且向控件发送绘制的消息; void Update(); //使控件重绘其工作区内的无效区域; void Refresh(); //相当于 this.invalidate(true); this.Update(); 双缓冲技术: 当数据量很大时,绘图可能需要几秒钟甚至更长的时间,而且有时还会出现闪烁现象,为了解决这些问题,可采用