溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++怎么實現(xiàn)Qt5雙緩沖機(jī)制

發(fā)布時間:2022-03-28 14:05:18 來源:億速云 閱讀:294 作者:iii 欄目:大數(shù)據(jù)

這篇文章主要講解了“C++怎么實現(xiàn)Qt5雙緩沖機(jī)制”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“C++怎么實現(xiàn)Qt5雙緩沖機(jī)制”吧!

1. 雙緩沖機(jī)制

所謂雙緩沖機(jī)制,是指在繪制控件時,首先將要繪制的內(nèi)容繪制在一個圖片中,再將圖片一次性地繪制到控件上。

在早期的Qt版本中,若直接在控件上進(jìn)行繪制工作,則在控件重繪時會產(chǎn)生閃爍的現(xiàn)象,控件重繪頻繁時,閃爍尤為明顯。

雙緩沖機(jī)制可以有效地消除這種閃爍現(xiàn)象。自Qt 5版本之后,QWidget 控件已經(jīng)能夠自動處理閃爍的問題。

因此,在控件上直接繪圖時,不用再操心顯示的閃爍問題,但雙緩沖機(jī)制在很多場合仍然有其用武之地。當(dāng)所需繪制的內(nèi)容較復(fù)雜并需要頻繁刷新,或者每次只需要刷新整個控件的一小部分時,仍應(yīng)盡量采用雙緩沖機(jī)制。

2. 實例

2.1 介紹

實現(xiàn)一個簡單的繪圖工具,可以選擇線形,線寬,顏色等基本要素

效果圖

C++怎么實現(xiàn)Qt5雙緩沖機(jī)制

2.2 部分關(guān)鍵代碼講解

構(gòu)造函數(shù)

DrawWidget::DrawWidget(QWidget *parent) :
    QWidget(parent)
{
    setAutoFillBackground(true);    //對窗體背景色的設(shè)置
    setPalette(QPalette(Qt::red));
    pix =new QPixmap(size());      	//此QPixmap對象用來準(zhǔn)備隨時接收繪制的內(nèi)容
    pix->fill(Qt::white);           //填充背景色為白色
    setMinimumSize(600,400);      	//設(shè)置繪制區(qū)窗體的最小尺寸
}

autoFillBackground

此屬性保存小部件背景是否自動填充

如果啟用,該屬性將導(dǎo)致Qt在調(diào)用paint事件之前填充小部件的背景。使用的顏色是由小部件調(diào)色板中的QPalette::Window顏色角色定義的。

此外,Windows總是填充QPalette::Window,除非設(shè)置了WA_OpaquePaintEvent或WA_NoSystemBackground屬性。

如果小部件的父組件有一個靜態(tài)背景漸變,則不能關(guān)閉這個屬性(即設(shè)置為false)。

void DrawWidget::mousePressEvent(QMouseEvent *e)
{
    startPos = e->pos();
}

重定義鼠標(biāo)按下事件 mousePressEvent(),在按下鼠標(biāo)按鍵時,記錄當(dāng)前的鼠標(biāo)位置值startPos。


重定義鼠標(biāo)移動事件mouseMoveEvent(),鼠標(biāo)移動事件在默認(rèn)情況下,在鼠標(biāo)按鍵被按下的同時拖曳鼠標(biāo)時被觸發(fā)。
QWidget的mouseTracking屬性指示窗體是否追蹤鼠標(biāo),默認(rèn)為 false(不追蹤),即在至少有一個鼠標(biāo)按鍵被按下的前提下移動鼠標(biāo)才觸發(fā)mouseMoveEvent()事件,可以通過setMouseTracking(bool enable)方法對該屬性值進(jìn)行設(shè)置。如果設(shè)置為追蹤,則無論鼠標(biāo)按鍵是否被按下,只要鼠標(biāo)移動,就會觸發(fā)mouseMoveEvent()事件。在此事件處理函數(shù)中,完成向QPixmap對象中繪圖的工作。具體代碼如下:

void DrawWidget::mouseMoveEvent(QMouseEvent *e)
{
    QPainter *painter = new QPainter;

    QPen pen;
    pen.setStyle((Qt::PenStyle)style);
    pen.setWidth(weight);
    pen.setColor(color);

    painter->begin(pix);
    painter->setPen(pen);
    painter->drawLine(startPos,e->pos());
    painter->end();
    startPos =e->pos();
    update();
}

三個set就不說了,大家都明白,說下begin

bool QPainter::begin(QPaintDevice **device*)

開始繪制繪制設(shè)備,如果成功返回true;否則返回false,這里是在Pixmap中繪圖

接下來是設(shè)置筆,然后看看drawLine函數(shù)

void QPainter::drawLine(const QPoint &p1, const QPoint &p2)

這是一個重載函數(shù)。從p1到p2畫一條線。

然后設(shè)置當(dāng)前的位置,e->pos()


看這個函數(shù)

void DrawWidget::paintEvent(QPaintEvent *)
{
    QPainter painter(this);
    painter.drawPixmap(QPoint(0,0),*pix);
}

這里是實現(xiàn)雙緩沖區(qū)域的地方

在上一個函數(shù)里,我們不是直接在面版上畫畫,而且在Pixmap里面畫畫,在這里,我們調(diào)用drawPixmap()函數(shù),將用于接收圖形繪制的QPixmap對象繪制在繪制區(qū)窗體控件上,這樣就實現(xiàn)了雙緩沖機(jī)制

void DrawWidget::resizeEvent(QResizeEvent *event)
{
    if(height()>pix->height()||width()>pix->width())
    {
        QPixmap *newPix = new QPixmap(size());
        newPix->fill(Qt::white);
        QPainter p(newPix);
        p.drawPixmap(QPoint(0,0),*pix);
        pix = newPix;
    }
    QWidget::resizeEvent(event);
}

調(diào)整繪制區(qū)大小函數(shù)resizeEvent(),當(dāng)窗體的大小發(fā)生改變時,效果看起來雖然像是繪制區(qū)大小改變了,但實際能夠進(jìn)行繪制的區(qū)域仍然沒有改變。因為繪圖的大小并沒有改變,還是原來繪制區(qū)窗口的大小,所以在窗體尺寸變化時應(yīng)及時調(diào)整用于繪制的QPixmap對象的大小。

最后一句QWidget::resizeEvent(event);是為了完成其余的工作


接下來實現(xiàn)clear函數(shù),

clear()函數(shù)完成繪制區(qū)的清除工作,只需調(diào)用一個新的、干凈的QPixmap對象來代替pix,并調(diào)用update()函數(shù)重繪即可。

void DrawWidget::clear()
{
    QPixmap *clearPix =new QPixmap(size());
    clearPix->fill(Qt::white);
    pix = clearPix;
    update();
}

看看被我們忽視的fill()函數(shù)

void QPixmap::fill(const QColor &color = Qt::white)

用給定的顏色填充像素圖。當(dāng)pixmap被繪制時,這個函數(shù)的效果是未定義的。

上期已經(jīng)說過的update()

更新小部件,除非禁用更新或隱藏小部件。

此函數(shù)不會導(dǎo)致立即重繪;相反,當(dāng)Qt返回到主事件循環(huán)時,它會安排一個油漆事件進(jìn)行處理。與調(diào)用repaint()相比,這允許Qt進(jìn)行優(yōu)化,以獲得更快的速度和更少的閃爍。

感謝各位的閱讀,以上就是“C++怎么實現(xiàn)Qt5雙緩沖機(jī)制”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對C++怎么實現(xiàn)Qt5雙緩沖機(jī)制這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點(diǎn)的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI