QT:使用“状态模式”绘制界面

时间:2021-09-01 00:57:55

QT与很多GUI库不同(如MFC),它不能随时随地地在界面上画图,只能在界面类的painterEvent中画图,如此一来,想在绘制QT界面时使用状态模式(GOF的23种设计模式之一)就有点困难了,作为解决方案,我先把要界面上的图片绘制在一张图片上(QPixmap),然后再在painterEvent中将Pixmap“画”到界面上。以下是这种方法的一个小例子。

截图:

QT:使用“状态模式”绘制界面QT:使用“状态模式”绘制界面QT:使用“状态模式”绘制界面

源代码:

  1. #include <QtGui>
  2. //状态类的基类,定义了各个公用接口,
  3. //其中,SetPixmap是一个纯虚接口
  4. class BasePen
  5. {
  6. protected:
  7. //这三个类成员理应是BasePen的私有成员,然后通过接口访问
  8. //我这里为了方便,直接把它们设为保护成员了
  9. QPixmap m_Pixmap;
  10. QPoint m_StartPoint;
  11. QPoint m_EndPoint;
  12. virtual void SetPixmap() = 0;
  13. public:
  14. BasePen()
  15. {
  16. m_StartPoint = m_EndPoint = QPoint(0, 0);
  17. m_Pixmap = QPixmap(500, 500);
  18. }
  19. void SetStartPoint(QPoint point) { m_StartPoint = point; }
  20. void SetEndPoint(QPoint point)
  21. {
  22. m_EndPoint = point;
  23. SetPixmap();
  24. }
  25. QPixmap GetPixmap() { return m_Pixmap; }
  26. };
  27. //矩形类,在界面上画一个红色的矩形
  28. class RectPen : public BasePen
  29. {
  30. protected:
  31. void SetPixmap()
  32. {
  33. m_Pixmap.fill(Qt::white);
  34. QPainter painter(&m_Pixmap);
  35. QRect rect(m_StartPoint, m_EndPoint);
  36. painter.setPen(Qt::red);
  37. painter.drawRect(rect);
  38. }
  39. };
  40. //直线类,在界面上画一条蓝色的直线
  41. class LinePen : public BasePen
  42. {
  43. protected:
  44. void SetPixmap()
  45. {
  46. m_Pixmap.fill(Qt::white);
  47. QPainter painter(&m_Pixmap);
  48. painter.setPen(Qt::blue);
  49. painter.drawLine(m_StartPoint, m_EndPoint);
  50. }
  51. };
  52. //圆形类,在界面上画一个绿色的椭圆
  53. class CirclePen : public BasePen
  54. {
  55. protected:
  56. void SetPixmap()
  57. {
  58. m_Pixmap.fill(Qt::white);
  59. QPainter painter(&m_Pixmap);
  60. QRect rect(m_StartPoint, m_EndPoint);
  61. painter.setPen(Qt::green);
  62. painter.drawEllipse(rect);
  63. }
  64. };
  65. class Widget : public QWidget
  66. {
  67. Q_OBJECT
  68. private:
  69. bool m_MouseDown;
  70. BasePen *m_BasePen;
  71. RectPen *m_RectPen;
  72. LinePen *m_LinePen;
  73. CirclePen *m_CirclePen;
  74. //在界面上放三个按钮,用来控制画图状态
  75. QRadioButton *m_LineButton;
  76. QRadioButton *m_RectButton;
  77. QRadioButton *m_CircleButton;
  78. protected:
  79. void mousePressEvent(QMouseEvent *event);
  80. void mouseMoveEvent(QMouseEvent *event);
  81. void mouseReleaseEvent(QMouseEvent *event);
  82. void paintEvent(QPaintEvent *event);
  83. public:
  84. Widget(QWidget *parent = 0);
  85. ~Widget();
  86. private slots:
  87. void ClickedLineButton() { m_BasePen = m_LinePen; }
  88. void ClickedRectButton() { m_BasePen = m_RectPen; }
  89. void ClickedCircleButton() { m_BasePen = m_CirclePen; }
  90. };
  91. Widget::Widget(QWidget *parent /* = 0 */)
  92. : QWidget(parent)
  93. {
  94. m_MouseDown = false;
  95. m_RectPen = new RectPen;
  96. m_LinePen = new LinePen;
  97. m_CirclePen = new CirclePen;
  98. m_LineButton = new QRadioButton("Line", this);
  99. m_RectButton = new QRadioButton("Rect", this);
  100. m_CircleButton = new QRadioButton("Circle", this);
  101. m_LineButton->move(10, 10);
  102. m_RectButton->move(100, 10);
  103. m_CircleButton->move(200, 10);
  104. connect(m_LineButton, SIGNAL(clicked()), this, SLOT(ClickedLineButton()));
  105. connect(m_RectButton, SIGNAL(clicked()), this, SLOT(ClickedRectButton()));
  106. connect(m_CircleButton, SIGNAL(clicked()), this, SLOT(ClickedCircleButton()));
  107. m_BasePen = m_LinePen;
  108. m_LineButton->setChecked(true);
  109. setFixedSize(500, 500);
  110. }
  111. Widget::~Widget()
  112. {
  113. delete m_LinePen;
  114. delete m_RectPen;
  115. delete m_CirclePen;
  116. }
  117. void Widget::mousePressEvent(QMouseEvent *event)
  118. {
  119. if( event->button() == Qt::LeftButton )
  120. {
  121. m_MouseDown = true;
  122. m_BasePen->SetStartPoint(event->pos());
  123. }
  124. }
  125. void Widget::mouseMoveEvent(QMouseEvent *event)
  126. {
  127. if( m_MouseDown )
  128. {
  129. m_BasePen->SetEndPoint(event->pos());
  130. update();
  131. }
  132. }
  133. void Widget::mouseReleaseEvent(QMouseEvent *event)
  134. {
  135. if( event->button() == Qt::LeftButton )
  136. {
  137. m_MouseDown = false;
  138. }
  139. }
  140. void Widget::paintEvent(QPaintEvent *event)
  141. {
  142. QPixmap temp = m_BasePen->GetPixmap();
  143. QPainter painter(this);
  144. painter.drawPixmap(0, 0, temp);
  145. }
  146. #include "main.moc"
  147. int main(int argc, char **argv)
  148. {
  149. QApplication app(argc, argv);
  150. Widget *ww = new Widget;
  151. ww->show();
  152. return app.exec();
  153. }

http://blog.csdn.net/small_qch/article/details/7632226