QT通过起点、终点、弧度(方向)来绘制圆弧

2023-11-02 07:10

本文主要是介绍QT通过起点、终点、弧度(方向)来绘制圆弧,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

时间:2021-8-17

为了可以让自己使用起点、终点和弧度(方向)来直接绘制圆弧,我准备自己开发一个绘制圆弧的函数。在网上查了很多资料,并没有查到自己想要的。

1、说明

这里的起点、终点指的是圆弧通过的两点。而弧度方向指的是圆弧起点弧线的“切线”方向,下面用角度θ来表示,逆时针方向为正,顺时针方向为负。

 如上图所示,已知P1、P2平面坐标和∠θ,需要利用QT绘图工具绘制出这一段圆弧。用户只需要输入P1、P2、P3三点即可,其中P3只要方向在直线P1P3方向上即可。

需要求出:

①  圆心C点坐标和半径R,其中R=CP1和CP2线段长度;

②  角度∠P1-C-X、∠P2-C-X的角度,可以看出角之间相差2θ度。

2、算法如下:

点M的坐标:(P1+P2)/2,P1P2和水平线的夹角角度=argtan((P2.y-P1.y)/(P2.x-P1.x)),设为ð

设P1-M的长度= L,则 半径R=P1-C长度=L/sinθ

C.x = P1.x + R / cos(90 - θ - ð)

C.y = P1.y + R / sin(90 - θ - ð) 即可得到C点坐标。

得到C点坐标后,易得,以C点为中心,边长为2R的矩形(该矩形即为QT所要求矩形)。

通过P1-C坐标求出 ∠P1-C-x即为圆弧起始角,易得∠P2-C-x角度。即为QT绘制圆弧的两个角度。

最后调用QT绘图方法即可。

3、QT程序代码

3.1 QT 绘图命令简介:

void QPainter::drawArc(const QRectF &rectangle, int startAngle, int spanAngle)

Draws the arc defined by the given rectangle, startAngle and spanAngle.

The startAngle and spanAngle must be specified in 1/16th of a degree, i.e. a full circle equals 5760 (16 * 360). Positive values for the angles mean counter-clockwise while negative values mean the clockwise direction. Zero degrees is at the 3 o'clock position.

以上命令主要内容为需要通过一个给定的矩形和起始角、偏移角来绘制圆弧。起始角和偏移角必须必须是度数的十六分之一。即1°表示为1*16,直角90°表示为90*16。举例如下:

  QRectF rectangle(10.0, 20.0, 80.0, 60.0);    //给定的矩形,我们会直接用正方形来绘制正圆弧int startAngle = 30 * 16;    //起始角,以3点方向为0°,逆时针为整。int spanAngle = 120 * 16;    //偏移角,就是起始角和终止角的差值。绘制整个圆就输入360 * 16QPainter painter(this);painter.drawArc(rectangle, startAngle, spanAngle);    //调用绘图命令

上面例子绘出的圆弧为:

3.2 我们的程序如下,程序变量可能与上述代号不同,请注意体会

int Helper::caculateArcParameters(arcParameters * arcPara, inputParameters inputPara)
{//返回struct arcParameters指针,使用指针而不是直接返回arcParameters变量仅仅是因为可以判断是否为NULL。//判断是否可以做圆弧,否则返回空NULL//角度为0, 或者两点坐标重合,返回NULLdouble x1 = inputPara.x1;double y1 = inputPara.y1;double x2 = inputPara.x2;double y2 = inputPara.y2;double jiaoDu = inputPara.jiaoDu; //本段弧线的弧度或角度 θint yinYong = inputPara.yinYong;if ((jiaoDu == 0)or(yinYong != 0)or((x1 == x2)and(y1 == y2))){return 1; //当没有弧度,或者有引用时,或者两点坐标重合时,返回1。正确返回0,如下。}else{
//        double mx = (x1+x2)/2; double my = (y1+y2)/2;double L = qSqrt((x2-x1)*(x2-x1) + (y2-y1)*(y2-y1));    //两点之间距离。调试正确double R = qAbs((0.5 * L) / qSin(jiaoDu));    //圆弧所在圆的半径  qsin都是弧度,角度需要转换。double orginJiaoDu = qAtan((y2-y1)/(x2-x1));    //起点终点组成的角度ð(单位弧度)double C_x = x1 + R / qCos(qDegreesToRadians(90.0) - jiaoDu + orginJiaoDu);  //圆心坐标xdouble C_y = y1 + R / qSin(qDegreesToRadians(90.0) - jiaoDu + orginJiaoDu);   //弧度相互加减qDebug()<<"orginjiaoDu:"<<orginJiaoDu<<" jiaodu:"<<jiaoDu;qDebug()<<"\n R:"<<R;qDebug()<<"\n C_x:"<<C_x<<"  C_y:"<<C_y;arcPara->rectParameter = QRectF(C_x - R, C_y - R, 2 * R, 2 * R);arcPara->startAngleParameter = qRadiansToDegrees(qAtan((y2 - C_y)/(x2 - C_x))) * 16; //y方向取负值arcPara->spanAngleParameter = arcPara->startAngleParameter - qRadiansToDegrees(2 * jiaoDu) * 16;
//        arcPara->spanAngleParameter = 360 * 16;return 0;}
}

3.3 以上程序还在调试中,希望可以对大家啊有所帮助。

函数通过一个结构指针返回计算结果,通过一个结构传输数据,结构定义如下:

truct arcParameters
{QRectF rectParameter;int startAngleParameter;int spanAngleParameter;
};
struct inputParameters
{double x1;double y1;double x2;double y2;double jiaoDu;int yinYong;
};

3.4 程序运行效果

这个程序的运行效果如上,上面图形可以根据下面输入数据来绘图。 来定制绘图。

这么做的目的是通过文本来保存图形。

注,程序还在优化中。

4. 结束

4.1 经过不屑的努力,程序终于能正常运算了。主要困难再坐标转换和三角函数中出现的基础错误。整体思路没有问题。

最主要的函数如下,如果对您有帮助,还请多支持。

int Helper::caculateArcParameters(arcParameters * arcPara, inputParameters inputPara)
{//返回struct arcParameters指针,使用指针而不是直接返回arcParameters变量仅仅是因为可以判断是否为NULL。//判断是否可以做圆弧,否则返回空NULL//角度为0, 或者两点坐标重合,返回NULL//[1]输入数据为x-y坐标,角度不变(逆时针为正),并判断输入数据是否合理。double x1 = inputPara.x1;double y1 = inputPara.y1;double x2 = inputPara.x2;double y2 = inputPara.y2;double radianOfDirect = inputPara.direct; //本段弧线的弧度 θ,表示起点方向偏离P1P2方向的角度,逆时针为正。int id = inputPara.idOfRefer;const double radianOf90 = qDegreesToRadians(90.0);if ((radianOfDirect == 0)or(qAbs(radianOfDirect) >= radianOf90 * 2)){return 1;   //大于180度  = 0 时退出确保方向位于 (0--180)之间,不含0和180度。};if ((id != 0)or((x1 == x2)and(y1 == y2))){return 1;}; //退出函数, 否则继续下面语句//[2] 计算P1、P2和X轴角度angleOfP1P2、中点M坐标及向量(M,P1),(M,P2)长度; 和半径radiusdouble angleOfP1P2 = qDegreesToRadians(getAngle((x2 - x1),(y2 - y1)));double m_x = (x1+x2)/2;double m_y = (y1+y2)/2;
//    double vectorMp2_x = x2 - m_x;
//    double vectorMp2_y = y2 - m_y;double lengthOfMP2 = qSqrt((x2 - m_x)*(x2 - m_x) + (y2 - m_y)*(y2 - m_y));    //两点之间距离。调试正确double radius = qAbs(lengthOfMP2 / sin(radianOfDirect));double vectorP2N_x; //P2指向圆心的单位长度向量double vectorP2N_y;double angleOfStart;double angleOfSpan;double C_x = 0;double C_y = 0;//[3]根据角度正负,优弧劣弧分别计算圆心坐标C_x C_y。先算单位P2指向圆心的向量再乘以半径长度即可。if (radianOfDirect > 0){//角度大于0vectorP2N_x = cos(angleOfP1P2)*cos(-radianOfDirect - radianOf90) - sin(angleOfP1P2)*sin(-radianOfDirect - radianOf90);vectorP2N_y = sin(angleOfP1P2)*cos(-radianOfDirect - radianOf90) + cos(angleOfP1P2)*sin(-radianOfDirect - radianOf90);}else if (radianOfDirect < 0){//角度小于0vectorP2N_x = cos(angleOfP1P2)*cos(-radianOfDirect + radianOf90) - sin(angleOfP1P2)*sin(-radianOfDirect + radianOf90);vectorP2N_y = sin(angleOfP1P2)*cos(-radianOfDirect + radianOf90) + cos(angleOfP1P2)*sin(-radianOfDirect + radianOf90);}else{return 1;//应该不会执行。};//所有情况C坐标均等于P2 + P2N向量乘以倍率!C_x = x2 + vectorP2N_x * radius;C_y = y2 + vectorP2N_y * radius;qDebug()<<"P1P2-X角度:"<<angleOfP1P2<<" 圆弧弧度方向:"<<radianOfDirect;qDebug()<<"半径:"<<radius;qDebug()<<"C_x:"<<C_x<<"  C_y:"<<C_y;qDebug()<<"x1,y1:"<<x1<<"__"<<y1;qDebug()<<"x2,y2:"<<x2<<"__"<<y2;//[4]根据角度,分别计算起始角度和偏转角度
//    angleOfStart = 0;
//    angleOfSpan = 90;// 验证的确时再“第四”象限。/* !!!坐标改为普通X-Y坐标后,角度的起始角度也有所变化。此时角度时逆时针旋转的!!!  */if (radianOfDirect > 0){//角度大于0,弧度向P1P2的左边弯曲,此时圆弧从P2到P1(顺时针)。angleOfStart = 360-getAngle((x1 - C_x), (y1 - C_y));// * 16;angleOfSpan = 360-getAngle((x2 - C_x), (y2 - C_y)) - angleOfStart;// * 16;}else if (radianOfDirect < 0){//角度小于0,与上面相反。圆弧为P1到P2angleOfStart = 360-getAngle((x2 - C_x), (y2 - C_y));// * 16;angleOfSpan = 360-getAngle((x1 - C_x), (y1 - C_y)) - angleOfStart;// * 16;}else{return 1;//应该不会执行。};//[5]把坐标Y值取负后,转换为y-x坐标,赋值给arcPara,返回0arcPara->rect = QRectF(C_x - radius, C_y - radius, 2 * radius, 2 * radius);qDebug()<<"StartAngle, SpanAngle:"<<angleOfStart<<"  "<<angleOfSpan;//    angleOfStart = 340;
//    angleOfSpan = 50;if (angleOfSpan < 0) angleOfSpan += 360;arcPara->angleOfStart = angleOfStart * 16;arcPara->angleOfSpan = angleOfSpan * 16;return 0;
};double Helper::getAngle(double x, double y) {double l = qSqrt(x*x + y*y);double a = qAcos(x/l);double ret = a * 180 / qDegreesToRadians(180.0); //弧度转角度,方便调试if (y < 0) {return 360 - ret;}return ret;
};

4.2 最终结果截图如下:

可以看出,不论P1 P2起始角如何变化,总能正确的绘制出圆弧。

 

这篇关于QT通过起点、终点、弧度(方向)来绘制圆弧的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/328976

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【WebGPU Unleashed】1.1 绘制三角形

一部2024新的WebGPU教程,作者Shi Yan。内容很好,翻译过来与大家共享,内容上会有改动,加上自己的理解。更多精彩内容尽在 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信号:digital_twin123 在 3D 渲染领域,三角形是最基本的绘制元素。在这里,我们将学习如何绘制单个三角形。接下来我们将制作一个简单的着色器来定义三角形内的像素

Flutter 进阶:绘制加载动画

绘制加载动画:由小圆组成的大圆 1. 定义 LoadingScreen 类2. 实现 _LoadingScreenState 类3. 定义 LoadingPainter 类4. 总结 实现加载动画 我们需要定义两个类:LoadingScreen 和 LoadingPainter。LoadingScreen 负责控制动画的状态,而 LoadingPainter 则负责绘制动画。

嵌入式方向的毕业生,找工作很迷茫

一个应届硕士生的问题: 虽然我明白想成为技术大牛需要日积月累的磨练,但我总感觉自己学习方法或者哪些方面有问题,时间一天天过去,自己也每天不停学习,但总感觉自己没有想象中那样进步,总感觉找不到一个很清晰的学习规划……眼看 9 月份就要参加秋招了,我想毕业了去大城市磨练几年,涨涨见识,拓开眼界多学点东西。但是感觉自己的实力还是很不够,内心慌得不行,总怕浪费了这人生唯一的校招机会,当然我也明白,毕业

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点

理解分类器(linear)为什么可以做语义方向的指导?(解纠缠)

Attribute Manipulation(属性编辑)、disentanglement(解纠缠)常用的两种做法:线性探针和PCA_disentanglement和alignment-CSDN博客 在解纠缠的过程中,有一种非常简单的方法来引导G向某个方向进行生成,然后我们通过向不同的方向进行行走,那么就会得到这个属性上的图像。那么你利用多个方向进行生成,便得到了各种方向的图像,每个方向对应了很多

使用Qt编程QtNetwork无法使用

使用 VS 构建 Qt 项目时 QtNetwork 无法使用的问题 - 摘叶飞镖 - 博客园 (cnblogs.com) 另外,强烈建议在使用QNetworkAccessManager之前看看这篇文章: Qt 之 QNetworkAccessManager踏坑记录-CSDN博客 C++ Qt开发:QNetworkAccessManager网络接口组件 阅读目录 1.1 通用API函数

Qt多语种开发教程

Qt作为跨平台的开发工具,早已应用到各行各业的软件开发中。 今天讲讲,Qt开发的正序怎么做多语言开发。就是说,你设置中文,就中文显示;设置英语就英文显示,设置繁体就繁体显示,设置发育就显示法语等。 开发环境(其实多语种这块根环境没太大关系):win10,Qt.5.12.10 一.先用QtCreator创建一个简单的桌面程序 1.工程就随便命名“LanguageTest”,其他默认。 2.在设计师