03.05 QT变乱

打印 上一主题 下一主题

主题 957|帖子 957|积分 2871

实现一个绘图工具,具备以下功能:



  • 鼠标绘制线条。
  • 及时调解线条颜色和粗细。
  • 橡皮擦功能,覆盖绘制内容。
  • 撤销功能,ctrl + z 快捷键撤销末了一笔
程序代码:

<1> Widget.h:

  1. #ifndef WIDGET_H
  2. #define WIDGET_H
  3. #include <QWidget>
  4. #include <QPaintEvent>
  5. #include <QPainter>
  6. #include <QDebug>
  7. #include <QMouseEvent>
  8. #include <QLine>
  9. #include <QVector>
  10. #include <QColorDialog>
  11. #include <QPair>
  12. #include <QKeyEvent>
  13. QT_BEGIN_NAMESPACE
  14. namespace Ui { class Widget; }
  15. QT_END_NAMESPACE
  16. class Widget : public QWidget
  17. {
  18.     Q_OBJECT
  19. public:
  20.     Widget(QWidget *parent = nullptr);
  21.     ~Widget();
  22. private:
  23.     Ui::Widget *ui;
  24.     QPoint start;
  25.     QPoint end;
  26.     QVector<QVector<QPair<QLine, QPair<QColor, int>>>> strokes; // 存储每一笔的所有线段
  27.     QVector<QPair<QLine, QPair<QColor, int>>> currentStroke;    // 当前正在绘制的一笔
  28.     QColor currentColor = Qt::black;                            // 当前颜色
  29.     int currentWidth = 1;                                        // 当前线条粗细
  30.     bool isEraserMode = false;                                   // 是否为橡皮擦模式
  31. protected:
  32.     virtual void paintEvent(QPaintEvent *event) override;
  33.     virtual void mouseMoveEvent(QMouseEvent *event) override;
  34.     virtual void mousePressEvent(QMouseEvent *event) override;
  35.     virtual void mouseReleaseEvent(QMouseEvent *event) override;
  36.     virtual void keyPressEvent(QKeyEvent *event) override; // 键盘事件处理
  37. private slots:
  38.     void on_pushButton_clicked(); // 调色板
  39.     void on_pushButton_2_clicked(); // 1mm
  40.     void on_pushButton_3_clicked(); // 5mm
  41.     void on_pushButton_4_clicked(); // 10mm
  42.     void on_pushButton_6_clicked(); // 20mm
  43.     void on_pushButton_5_clicked(); // 橡皮擦
  44. };
  45. #endif // WIDGET_H
复制代码
<2> Widget.cpp:

  1. #include "widget.h"
  2. #include "ui_widget.h"
  3. Widget::Widget(QWidget *parent)
  4.     : QWidget(parent)
  5.     , ui(new Ui::Widget)
  6. {
  7.     ui->setupUi(this);
  8.     setPalette(QPalette(Qt::white)); // 设置窗口背景色为白色
  9.     setAutoFillBackground(true);     // 自动填充背景
  10. }
  11. Widget::~Widget()
  12. {
  13.     delete ui;
  14. }
  15. void Widget::paintEvent(QPaintEvent *event)
  16. {
  17.     QPainter painter(this);
  18.     QPen pen;
  19.     // 遍历所有笔触,绘制每一笔
  20.     for (const auto &stroke : strokes) {
  21.         for (const auto &linePair : stroke) {
  22.             pen.setColor(linePair.second.first); // 设置线条颜色
  23.             pen.setWidth(linePair.second.second); // 设置线条粗细
  24.             painter.setPen(pen);
  25.             painter.drawLine(linePair.first); // 绘制线条
  26.         }
  27.     }
  28.     // 绘制当前正在绘制的一笔
  29.     for (const auto &linePair : currentStroke) {
  30.         pen.setColor(linePair.second.first); // 设置线条颜色
  31.         pen.setWidth(linePair.second.second); // 设置线条粗细
  32.         painter.setPen(pen);
  33.         painter.drawLine(linePair.first); // 绘制线条
  34.     }
  35. }
  36. void Widget::mouseMoveEvent(QMouseEvent *event)
  37. {
  38.     if (event->buttons() & Qt::LeftButton) {
  39.         end = event->pos();
  40.         QLine line(start, end);
  41.         // 如果是橡皮擦模式,使用背景色绘制
  42.         if (isEraserMode) {
  43.             QColor backgroundColor = palette().color(QPalette::Window); // 获取窗口背景色
  44.             currentStroke << qMakePair(line, qMakePair(backgroundColor, currentWidth)); // 使用背景色覆盖
  45.         } else {
  46.             currentStroke << qMakePair(line, qMakePair(currentColor, currentWidth)); // 正常绘制
  47.         }
  48.         start = end;
  49.         update(); // 触发重绘
  50.     }
  51. }
  52. void Widget::mousePressEvent(QMouseEvent *event)
  53. {
  54.     if (event->button() == Qt::LeftButton) {
  55.         start = event->pos(); // 记录起点
  56.         currentStroke.clear(); // 开始新的一笔
  57.     }
  58. }
  59. void Widget::mouseReleaseEvent(QMouseEvent *event)
  60. {
  61.     if (event->button() == Qt::LeftButton) {
  62.         end = event->pos();
  63.         QLine line(start, end);
  64.         // 如果是橡皮擦模式,使用背景色绘制
  65.         if (isEraserMode) {
  66.             QColor backgroundColor = palette().color(QPalette::Window); // 获取窗口背景色
  67.             currentStroke << qMakePair(line, qMakePair(backgroundColor, currentWidth)); // 使用背景色覆盖
  68.         } else {
  69.             currentStroke << qMakePair(line, qMakePair(currentColor, currentWidth)); // 正常绘制
  70.         }
  71.         strokes << currentStroke; // 将当前一笔添加到所有笔触中
  72.         currentStroke.clear(); // 清空当前一笔
  73.         update(); // 触发重绘
  74.     }
  75. }
  76. // 键盘事件处理
  77. void Widget::keyPressEvent(QKeyEvent *event)
  78. {
  79.     // 检测 Ctrl + Z 按键组合
  80.     if (event->modifiers() == Qt::ControlModifier && event->key() == Qt::Key_Z) {
  81.         if (!strokes.isEmpty()) {
  82.             strokes.removeLast(); // 移除最后一整笔
  83.             update(); // 触发重绘
  84.         }
  85.     }
  86. }
  87. // 打开调色板
  88. void Widget::on_pushButton_clicked()
  89. {
  90.     currentColor = QColorDialog::getColor(currentColor, this, "选择颜色");
  91. }
  92. void Widget::on_pushButton_2_clicked()
  93. {
  94.     currentWidth = 1; // 设置线条粗细为 1
  95. }
  96. void Widget::on_pushButton_3_clicked()
  97. {
  98.     currentWidth = 5; // 设置线条粗细为 5
  99. }
  100. void Widget::on_pushButton_4_clicked()
  101. {
  102.     currentWidth = 10; // 设置线条粗细为 10
  103. }
  104. void Widget::on_pushButton_6_clicked()
  105. {
  106.     currentWidth = 20; // 设置线条粗细为 20
  107. }
  108. // 切换橡皮擦模式
  109. void Widget::on_pushButton_5_clicked()
  110. {
  111.     isEraserMode = !isEraserMode; // 切换橡皮擦模式
  112.     if (isEraserMode) {
  113.         ui->pushButton_5->setText("绘图模式"); // 更新按钮文本
  114.     } else {
  115.         ui->pushButton_5->setText("橡皮擦"); // 更新按钮文本
  116.     }
  117. }
复制代码
<3> main.cpp:

  1. #include "widget.h"
  2. #include <QApplication>
  3. int main(int argc, char *argv[])
  4. {
  5.     QApplication a(argc, argv);
  6.     Widget w;
  7.     w.show();
  8.     return a.exec();
  9. }
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

我爱普洱茶

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表