当前位置: 首页 > news >正文

QT实现单个控制点在曲线上的贝塞尔曲线

最终效果:
在这里插入图片描述
一共三个文件
main.cpp

#include <QApplication>
#include "SplineBoard.h"
int main(int argc,char** argv) {QApplication a(argc, argv);SplineBoard b;b.setWindowTitle("标准的贝塞尔曲线");b.show();SplineBoard b2(0.0001);b2.show();b2.setWindowTitle("控制点在曲线上的贝塞尔曲线");b.move(0,0);b2.move(800,0);return a.exec();
}

SplineBoard.h

#ifndef SPLINEBOARD_H
#define SPLINEBOARD_H
#include <QPainter>
#include <QWidget>
class SplineBoard:public QWidget
{Q_OBJECT
public:SplineBoard(double terminalRatio = 0.25);protected:void mouseMoveEvent(QMouseEvent* e);void mousePressEvent(QMouseEvent* e);void mouseReleaseEvent(QMouseEvent* e);void paintEvent(QPaintEvent* e);int mPressedPos;QPointF mStart;QPointF mEnd;QPointF mCont;const double termRatio;
};#endif // SPLINEBOARD_H

SplineBoard.cpp

#include "SplineBoard.h"
#include <QDebug>
#include <QMouseEvent>
#include <QPainterPath>static int radius = 8;
static const int PosEmpty = 0;
static const int PosStart = 1;
static const int PosCont = 2;
static const int PosEnd = 4;static void buildQuadBezier(QPainterPath& pp, const QPointF& p0,const QPointF& p1,const QPointF& p2, qreal term){pp.moveTo(p0);const qreal a = (term-0.5)*-4;const qreal d = (1-2*term)*-4;for (qreal t = 0; t < 1.01; t += 0.01) {qreal A = a*t*t+(-1-a)*t+1;qreal B = d*t*t - d*t;qreal C = a*t*t + (1-a)*t;qreal x = A * p0.x() + B * p1.x() + C * p2.x();qreal y = A * p0.y() + B * p1.y() + C * p2.y();pp.lineTo(x, y);}
}
SplineBoard::SplineBoard(double terminalRatio):mPressedPos(PosEmpty),termRatio(terminalRatio)
{resize(800,600);setWindowTitle("spline board");mStart = QPointF(50,300);mEnd = QPointF(750,300);mCont = QPointF(400,300);
}void SplineBoard::mouseMoveEvent(QMouseEvent* e){auto cur = e->pos();if(mPressedPos == PosStart){mStart = cur;}else if(mPressedPos == PosCont){mCont = cur;}else if(PosEnd == mPressedPos){mEnd = cur;}else{return;}update();
}
void SplineBoard::mousePressEvent(QMouseEvent* e){auto pos = e->pos();auto startRect = QRectF(mStart,QSize(radius*2,radius*2));startRect.translate(-radius,-radius);auto contRect = QRectF(mCont,QSize(radius*2, radius*2));contRect.translate(-radius,-radius);auto endRect = QRectF(mEnd , QSize(radius*2, radius*2));endRect.translate(-radius,-radius);if(startRect.contains(pos)){mPressedPos = PosStart;}else if(contRect.contains(pos)) {mPressedPos = PosCont;}else if(endRect.contains(pos)){mPressedPos = PosEnd;}else{mPressedPos = PosEmpty;}}
void SplineBoard::mouseReleaseEvent(QMouseEvent* e){mPressedPos = PosEmpty;
}void SplineBoard::paintEvent(QPaintEvent* e){QPainter painter(this);painter.setRenderHint(QPainter::Antialiasing);QPen pen(Qt::blue);pen.setWidth(2);painter.setPen(pen);QPainterPath path;buildQuadBezier(path,mStart,mCont,mEnd,termRatio);painter.drawPath(path);painter.setPen(Qt::NoPen);painter.setBrush(Qt::magenta);painter.drawEllipse(mStart,radius,radius);  // Draw start pointpainter.drawEllipse(mCont, radius,radius);  // Draw control pointpainter.drawEllipse(mEnd, radius,radius);  // Draw end pointQWidget::paintEvent(e);}
http://www.lryc.cn/news/547336.html

相关文章:

  • svn 通过127.0.01能访问 但通过公网IP不能访问,这是什么原因?
  • ‌学习DeepSeek V3 与 R1 核心区别(按功能维度分类)
  • C++中的 互斥量
  • 直接法估计相机位姿
  • PHP动态网站建设
  • 【gRPC】Java高性能远程调用之gRPC详解
  • 数据结构知识学习小结
  • 分布式锁—2.Redisson的可重入锁一
  • 计算机毕业设计SpringBoot+Vue.js球队训练信息管理系统(源码+文档+PPT+讲解)
  • FFmpeg入门:最简单的音视频播放器
  • java 查找两个集合的交集部分数据
  • 【系统架构设计师】以数据为中心的体系结构风格
  • 通过HTML有序列表(ol/li)实现自动递增编号的完整解决方案
  • 【Python 数据结构 4.单向链表】
  • 基于 vLLM 部署 LSTM 时序预测模型的“下饭”(智能告警预测与根因分析部署)指南
  • Java多线程与高并发专题——ConcurrentHashMap 在 Java7 和 8 有何不同?
  • NL2SQL-基于Dify+阿里通义千问大模型,实现自然语音自动生产SQL语句
  • LeetCode 1328.破坏回文串:贪心
  • 计算机视觉|ViT详解:打破视觉与语言界限
  • //定义一个方法,把int数组中的数据按照指定的格式拼接成一个字符串返回,调用该方法,并在控制台输出结果
  • Python快捷手册
  • QT5 GPU使用
  • 如何在Spring Boot中读取JAR包内resources目录下文件
  • 《张一鸣,创业心路与算法思维》
  • SSE 和 WebSocket 的对比
  • es如何进行refresh?
  • Kubespray部署企业级高可用K8S指南
  • 【实战篇】【深度解析DeepSeek:从机器学习到深度学习的全场景落地指南】
  • 优选算法的智慧之光:滑动窗口专题(二)
  • Kylin麒麟操作系统服务部署 | NFS服务部署