Qt鼠标拖动线条组成的多边形移动
创始人
2025-01-10 04:36:46
0

实现在鼠标按下线条时可以拖动整个图形的功能,可以按照以下步骤进行:

  1. 记录顶点:定义一个顶点列表,存储多边形的每个顶点。

  2. 检测鼠标按下:在mousePressEvent中,检测鼠标是否在多边形的任意一条边上。

  3. 记录按下的顶点:如果鼠标在线上,记录下被按下的顶点的索引。

  4. 计算偏移量:在鼠标按下时,计算鼠标位置和被按下顶点之间的偏移量。

  5. 移动顶点:在mouseMoveEvent中,如果鼠标按钮保持按下状态,则根据鼠标的当前位置和之前计算的偏移量更新顶点的位置。

  6. 重绘图形:在顶点移动后,调用update方法来请求重绘控件。

  7. 结束拖动:在mouseReleaseEvent中,结束拖动操作。

    #include  #include  #include  #include  #include  #include   class MovablePolygon : public QWidget {     Q_OBJECT  public:     MovablePolygon(QWidget *parent = nullptr) : QWidget(parent),         dragging(false), currentPointIndex(-1), dragOffset(0, 0) {         // 初始化多边形顶点         points << QPointF(50, 50) << QPointF(150, 50)                 << QPointF(150, 150) << QPointF(50, 150);     }  protected:     void paintEvent(QPaintEvent *event) override {         QPainter painter(this);         painter.setPen(QPen(Qt::black, 2));         for (int i = 0; i < points.size() - 1; ++i) {             painter.drawLine(points[i], points[i + 1]);         }         // 闭合多边形         painter.drawLine(points.last(), points.first());     }      void mousePressEvent(QMouseEvent *event) override {         if (event->button() == Qt::LeftButton) {             dragging = true;             for (int i = 0; i < points.size() - 1; ++i) {                 if (isPointOnLine(event->pos(), points[i], points[i + 1], 10)) {                     currentPointIndex = i;                     dragOffset = event->pos() - points[i];                     break;                 }             }             if (currentPointIndex == -1) {                 // 如果没有点被按下,可能需要移动整个多边形                 dragOffset = event->pos() - points[0];             }         }     }      void mouseMoveEvent(QMouseEvent *event) override {         if (dragging && event->buttons() & Qt::LeftButton) {             if (currentPointIndex != -1) {                 points[currentPointIndex] = event->pos() - dragOffset;             } else {                 // 移动整个多边形                 for (int i = 0; i < points.size(); ++i) {                     points[i] += event->pos() - dragOffset;                 }             }             update(); // 重绘控件         }     }      void mouseReleaseEvent(QMouseEvent *event) override {         if (event->button() == Qt::LeftButton) {             dragging = false;             currentPointIndex = -1;         }     }  private:     bool dragging;     int currentPointIndex;     QPoint dragOffset;     QVector points;      bool isPointOnLine(const QPointF &point, const QPointF &lineStart, const QPointF &lineEnd, double tolerance) {         double dist = qAbs((lineEnd.y() - lineStart.y()) * point.x()                             - (lineEnd.x() - lineStart.x()) * point.y()                             + lineEnd.x() * lineStart.y()                             - lineEnd.y() * lineStart.x());         double lineLength = QLineF(lineStart, lineEnd).length();         return dist <= tolerance && lineLength >= qAbs(point.x() - lineStart.x());     } };  #include "MovablePolygon.moc" // 确保包含moc文件

    在这个示例中,MovablePolygon类是一个自定义的QWidget,它包含一个多边形顶点列表points。我们重写了paintEvent来绘制多边形,重写了鼠标事件来处理拖动逻辑。isPointOnLine函数用于检测鼠标点击是否在多边形的边上,如果检测到鼠标在线上,就记录下顶点索引和偏移量,并在鼠标移动时更新顶点位置。

相关内容

热门资讯

透视存在!德州局脚本,拱趴大菠... 透视存在!德州局脚本,拱趴大菠萝辅助神器,教材教程(有挂实锤)-哔哩哔哩一、拱趴大菠萝辅助神器AI软...
十分钟软件!丫丫陕西插件,闲逸... 十分钟软件!丫丫陕西插件,闲逸免费app辅助,切实是有挂(有挂教程)-哔哩哔哩1、完成闲逸免费app...
透视模拟器!wepoker究竟... 透视模拟器!wepoker究竟有没有透视,wejoker辅助软件,演示教程(有挂实锤)-哔哩哔哩1、...
第8分钟方法!免费辅助神器ap... 第8分钟方法!免费辅助神器app,贪吃蛇辅助器2022,竟然有挂(有挂方法)-哔哩哔哩1、上手简单,...
透视透视挂!pokemmo辅助... 透视透视挂!pokemmo辅助器脚本下载,hhpoker有辅助的吗,演示教程(揭秘有挂)-哔哩哔哩1...
第九分钟插件!新广西老友辅助,... 第九分钟插件!新广西老友辅助,佛手在线辅助器苹果版,都是是有挂(有挂细节)-哔哩哔哩运佛手在线辅助器...
透视有挂!哈糖大菠萝可以开挂吗... 透视有挂!哈糖大菠萝可以开挂吗,wpk私人局有透视吗,举措教程(有挂详情)-哔哩哔哩暗藏猫腻,小编详...
九分钟教程!福建天天开心辅助工... 九分钟教程!福建天天开心辅助工具视频,拱趴游戏诀窍,竟然真的是有挂(存在有挂)-哔哩哔哩1)福建天天...
透视透视!wepoker轻量版... 透视透视!wepoker轻量版透视方法,wpk俱乐部有没有辅助,举措教程(有挂秘诀)-哔哩哔哩亲,关...
五分钟神器!约战破解,赣牌圈的... 五分钟神器!约战破解,赣牌圈的隐藏机制,本来是真的挂(有挂分享)-哔哩哔哩1、首先打开赣牌圈的隐藏机...