Qtダブルバッファの描画

転載:https://wizardforcel.gitbooks.io/qt-beginning/content/22.html

はじめに

前のセクションでは、簡単なグラフィティボードの実装方法について説明しましたが、今回は、グラフィティボードに描画グラフィックスを実装します。ここでは、例として長方形を使用します。ダブルバッファドローイングの概念については後で説明します。

環境:Windows Xp + Qt 4.8.4 + QtCreator 2.6.2

ディレクトリ

  • まず、長方形を描きます
  • 2、ダブルバッファドローイング

テキスト

まず、長方形を描きます

1。前のプログラムに基づいてまだ変更しpainEvent()ます。最初に関数を変更します。

 
  1. void Dialog::paintEvent(QPaintEvent *)

  2. {

  3. QPainter painter(this);

  4. int x,y,w,h;

  5. x = lastPoint.x();

  6. y = lastPoint.y();

  7. w = endPoint.x() - x;

  8. h = endPoint.y() - y;

  9. painter.drawRect(x, y, w, h);

  10. }

ここでは、lastPointとendPointの点を通過して描画される長方形の開始点、幅、高さを決定します。プログラムを実行し、マウスで長方形をドラッグします。効果は下図に示されています。

18-1.jpguploading.4e448015.gif失敗した転送と再アップロードのキャンセル

2。長方形は上からドラッグできますが、この方法でウィンドウに直接描くことができます。以前に描いた長方形は保存できません。下にキャンバスを追加して、キャンバスに描画します。paintEvent()この関数は、以下のように変更します。

 
  1. void Dialog::paintEvent(QPaintEvent *)

  2. {

  3. int x,y,w,h;

  4. x = lastPoint.x();

  5. y = lastPoint.y();

  6. w = endPoint.x() - x;

  7. h = endPoint.y() - y;

  8. QPainter pp(&pix);

  9. pp.drawRect(x, y, w, h);

  10. QPainter painter(this);

  11. painter.drawPixmap(0, 0, pix);

  12. }

ここでは、まずキャンバスにグラフィックスを描画し、次にキャンバスをウィンドウに描画します。プログラムを実行してから、マウスを使用して四角形をドラッグし、幽霊がたくさんあることを発見しました。効果を以下に示します。

18-2.jpguploading.4e448015.gif失敗した転送と再アップロードのキャンセル

なぜそのような現象があるのですか?マウスをドラッグすることで長方形をすばやく、マウスをドラッグすることでゆっくりと長方形を描くことができます。その結果、ドラッグ速度が速いほどゴーストが少なくなります。実際、マウスをドラッグする過程で画面が何度もリフレッシュされており、paintEvent()関数が何度も実行され、実行されるたびに長方形が描かれていることもわかります。その理由を知って、この問題を回避する方法があります。

2、ダブルバッファドローイング

1。別の補助キャンバスを追加します。描画している場合、つまり、マウスボタンが離されていない場合は、この補助キャンバスに描画します。マウスボタンが離されたときのみ、実際のキャンバスに描画します。

まずdialog.h、2つのプライベート変数をファイルに追加します。

 
  1. QPixmap tempPix; //辅助画布

  2. bool isDrawing; //标志是否正在绘图

次にdialog.cpp、コンストラクターで変数を初期化します。

isDrawing = false;

次に、paintEvent()関数を変更します。

 
  1. void Dialog::paintEvent(QPaintEvent *)

  2. {

  3. int x,y,w,h;

  4. x = lastPoint.x();

  5. y = lastPoint.y();

  6. w = endPoint.x() - x;

  7. h = endPoint.y() - y;

  8. QPainter painter(this);

  9. if(isDrawing) //如果正在绘图,就在辅助画布上绘制

  10. {

  11. //将以前pix中的内容复制到tempPix中,保证以前的内容不消失

  12. tempPix = pix;

  13. QPainter pp(&tempPix);

  14. pp.drawRect(x,y,w,h);

  15. painter.drawPixmap(0, 0, tempPix);

  16. } else {

  17. QPainter pp(&pix);

  18. pp.drawRect(x,y,w,h);

  19. painter.drawPixmap(0,0,pix);

  20. }

  21. }

また、マウスダウンイベントハンドラーとマウスリリースイベントハンドラーの内容を変更する必要もあります。

 
  1. void Dialog::mousePressEvent(QMouseEvent *event)

  2. {

  3. if(event->button()==Qt::LeftButton) //鼠标左键按下

  4. {

  5. lastPoint = event->pos();

  6. isDrawing = true; //正在绘图

  7. }

  8. }

  9. void Dialog::mouseReleaseEvent(QMouseEvent *event)

  10. {

  11. if(event->button() == Qt::LeftButton) //鼠标左键释放

  12. {

  13. endPoint = event->pos();

  14. isDrawing = false; //结束绘图

  15. update();

  16. }

  17. }

マウスの左ボタンを押すと、図面のマーキングが開始され、ボタンを放すと、図面のマーキングがキャンセルされます。ここでプログラムを実行すると、通常の描画を行うことができます。効果を以下に示します。

18-3.jpguploading.4e448015.gif失敗した転送と再アップロードのキャンセル

2。ダブルバッファドローイング

この例で使用されている手法に基づいて、いわゆるダブルバッファドローイングの概念を紹介します。ダブルバッファ(ダブルバッファ)描画、つまり、描画する場合、すべてのコンテンツが描画デバイス(などQPixmap)に描画され、次に画像全体がパーツに描画されて表示されます。ダブルバッファ描画を使用すると、表示中のちらつきを回避できます。Qt 4.0以降、QWidgetパーツのすべての描画は自動的にダブルバッファリングpaintEvent()を使用するため、通常、関数ダブルバッファリングコードを使用してちらつきを回避する必要はありません

一般的な描画では手動でダブルバッファ描画を使用する必要はありませんが、描画効果を実現するには、ダブルバッファの概念に依存する必要があります。たとえば、このプログラムでは、マウスを使用して、インターフェイス上に任意のサイズの長方形を描画します。ここでは2つのキャンバスが必要ですが、どちらもQPixmap例です。1つはtempPix一時バッファーとして使用されます。マウスで長方形をドラッグして描画すると、コンテンツが最初に描画tempPixtempPixれてからインターフェイスに描画され、もう1つはpixバッファーとして使用されます、完成した図面を保存するために使用されます。マウスtempPixを離して長方形の描画を完了すると、コンテンツがpixにコピーされます。描画時にスミアを表示せず、以前に描画されたコンテンツが消えないようにするために、マウスの移動中に描画tempPixするたびに、長方形の元の画像に描画する必要があるため毎回描画する必要があります。まず、pixのコンテンツをコピーしますtempPixここには2つのQPixmapオブジェクトがあるため、バッファが2つあると言え、ダブルバッファドローイングと呼ばれます。

おわりに

Qtの基本的な描画の内容については、ここで説明しますが、これらの基本的な知識をより体系的かつ詳細に学習したい場合は、「Qt Creatorのクイックスタート」の第10章を参照してください。次のセクションから、Qtのグラフィカルビューフレームワークを簡単に紹介します。

元の記事を85件公開 賞賛された18件 120,000回の閲覧

おすすめ

転載: blog.csdn.net/a1317338022/article/details/105534088