服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - Qt 实现钢笔画线效果示例及详细原理

Qt 实现钢笔画线效果示例及详细原理

2021-08-30 13:35luoyayun361 C/C++

这篇文章主要介绍了Qt 实现钢笔画线效果示例及详细原理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

前言

上一篇文章:Qt 实现画线笔锋效果详细原理,根据这篇介绍的实现笔锋效果的原理,我们很容易实现另外一种笔效:钢笔。

所谓的钢笔笔效,就是真实还原钢笔书写出来的线条效果,其特征就是:根据笔的绘制速度而线条的宽度会逐渐变化,写得越快,线条越细,并且在收笔时带有笔锋效果。

那么,在上一篇文章的基础上,稍微修改一下,就可以实现这个效果,看下效果图:

Qt 实现钢笔画线效果示例及详细原理

Qt 实现钢笔画线效果示例及详细原理

Qt 实现钢笔画线效果示例及详细原理

实现原理

从上一篇文章我们知道,绘制的曲线是通过每两个点形成一条贝塞尔曲线,所以在不松手的情况下连续画线,整条线段是包含很多条path组合而成的。而要实现钢笔效果的关键,是要让线条的粗细跟随绘制的速度来变化。

之前看过很多Android上实现钢笔或者毛笔的算法,都是需要计算去画线速度,根据速度来动态改变线条的粗细。但是我这里没有计算速度,而是直接通过每一段path的长度来计算一个合理的宽度值出来。

众所周知,长度(也就是距离)= 速度*时间,在单位时间内,速度和距离是成正比的,所以我们通过两点间的距离来做判断也是一样的,没多大区别,并且还不用单独去计算速度了,简单省事儿。

那么,我们要实现的效果是,画线速度越快线条会越细,而画线速度越快,所采集到的两点间的距离就会越大,而我们是通过两点间距离来做参考依据,也就是说,两点间距和线条粗细是成反比的,两点间距越长,对应的这条path就越细,距离越短,path就越粗,二者是线性关系。当然,这里path的宽度会有一个最大值和最小值,需要在实际的场景中进行调试。

OK,根据以上分析,我们可以得到以下的示意图:

Qt 实现钢笔画线效果示例及详细原理

每条path都是通过两个坐标点实时生成的贝塞尔曲线。
在绘制这条曲线的时候,先获取到曲线的长度,然后线性计算出一个宽度值。
如何获取path的长度呢?
这个好办,QPainterPath有自带的接口length()

Qt 实现钢笔画线效果示例及详细原理

计算曲线的宽度,我写了一个简单的计算方法:

  1. qreal WbCanvasItem::calPathWidth(QPainterPath path)
  2. {
  3. qreal length = path.length();
  4. qreal width = PENWIDTH;
  5.  
  6. qreal t = length/10. - 1;
  7. width = PENWIDTH - t;
  8. if(width < 3){ //最小宽度
  9. width = 3;
  10. }
  11. return width;
  12. }

PENWIDTH是一个宏定义,曲线最大宽度;

根据以上步骤,我们来看一下效果:

Qt 实现钢笔画线效果示例及详细原理

为了方便看效果,每条path用了不同的颜色来区分。我们可以很明显的看到,path的宽度是不一样的,并且每条path的连接处的宽度变化非常明显,那么要怎么使其连接处变得平滑呢?

这时候就要用到上一篇介绍的方法进行补点了。这里的补点比上一篇文章中说的稍微麻烦点,需要将中间那根线条的两头都要补充点,其原理是一样的。

看一下示意图:

Qt 实现钢笔画线效果示例及详细原理

以上红色圈圈部分,就是补充的点。
从以上图可以看到,path2是倒数第二条path,path3是最后一条path。

需要注意的是,图中补充的两个地方,并不是同一时间补充的,当有新的path到来,只需要判断最新的path和上一个path的宽度,从而决定是补充到上一个path还是当前最新的path上。

这段话有点拗口,拆解一下:

Qt 实现钢笔画线效果示例及详细原理

假如这里path2是最后的一条path,而path1是倒数第二条,判断出来path2宽度笔path1小,那么就在path2的路径上补充点;

再看一种情况:

Qt 实现钢笔画线效果示例及详细原理

同样,这里path2是最后的一条path,而path1是倒数第二条,判断出来path2宽度笔path1大,那么就在path1的路径上补充点;

这样描述就很容易理解了。

OK,我们看一下补充点的代码:

  1. void WbCanvasItem::drawPatchPoint2(QPainter *painter, QPainterPath lastPath,
  2. QPainterPath curPath,
  3. qreal lastWidth, qreal curWidth)
  4. {
  5. qreal tPatchLength = 100.;
  6. if(lastWidth < curWidth){
  7. tPatchLength = calPatchLength(curPath.length());
  8.  
  9. qreal temp = (curWidth-lastWidth)/tPatchLength;
  10. int k = 0;
  11.  
  12. for (qreal i = 1;i > (100-tPatchLength)/100.; i-=0.01) {
  13. k++;
  14. painter->setPen(QPen(Qt::black,curWidth-temp*k, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
  15. painter->drawPoint(lastPath.pointAtPercent(i));
  16. }
  17. }
  18. else if(lastWidth > curWidth){
  19.  
  20. tPatchLength = calPatchLength(curPath.length());
  21.  
  22. qreal temp = (lastWidth-curWidth)/tPatchLength;
  23. int k = 0;
  24.  
  25. for (qreal i = 0;i < tPatchLength/100.; i+=0.01) {
  26. k++;
  27. painter->setPen(QPen(Qt::black,lastWidth-temp*k, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
  28. painter->drawPoint(curPath.pointAtPercent(i));
  29. }
  30. }
  31. }

看一下补充点后的效果:

Qt 实现钢笔画线效果示例及详细原理

这里的黑色部分就是动态补充上去的点。

好了, 整理原理分析完成,其实和前面一篇文章原理差不多,只是多了一步判断距离然后计算线宽的过程。

到此这篇关于Qt 实现钢笔画线效果示例及详细原理的文章就介绍到这了,更多相关Qt 钢笔画线内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/luoyayun361/article/details/104340258

延伸 · 阅读

精彩推荐
  • C/C++C语言main函数的三种形式实例详解

    C语言main函数的三种形式实例详解

    这篇文章主要介绍了 C语言main函数的三种形式实例详解的相关资料,需要的朋友可以参考下...

    ieearth6912021-05-16
  • C/C++使用C++制作简单的web服务器(续)

    使用C++制作简单的web服务器(续)

    本文承接上文《使用C++制作简单的web服务器》,把web服务器做的功能稍微强大些,主要增加的功能是从文件中读取网页并返回给客户端,而不是把网页代码...

    C++教程网5492021-02-22
  • C/C++OpenCV实现拼接图像的简单方法

    OpenCV实现拼接图像的简单方法

    这篇文章主要为大家详细介绍了OpenCV实现拼接图像的简单方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    iteye_183805102021-07-29
  • C/C++关于C语言中E-R图的详解

    关于C语言中E-R图的详解

    今天小编就为大家分享一篇关于关于C语言中E-R图的详解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看...

    Struggler095962021-07-12
  • C/C++c/c++实现获取域名的IP地址

    c/c++实现获取域名的IP地址

    本文给大家汇总介绍了使用c/c++实现获取域名的IP地址的几种方法以及这些方法的核心函数gethostbyname的详细用法,非常的实用,有需要的小伙伴可以参考下...

    C++教程网10262021-03-16
  • C/C++C语言实现双人五子棋游戏

    C语言实现双人五子棋游戏

    这篇文章主要为大家详细介绍了C语言实现双人五子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    两片空白7312021-11-12
  • C/C++深入C++拷贝构造函数的总结详解

    深入C++拷贝构造函数的总结详解

    本篇文章是对C++中拷贝构造函数进行了总结与介绍。需要的朋友参考下...

    C++教程网5182020-11-30
  • C/C++c/c++内存分配大小实例讲解

    c/c++内存分配大小实例讲解

    在本篇文章里小编给大家整理了一篇关于c/c++内存分配大小实例讲解内容,有需要的朋友们可以跟着学习参考下。...

    jihite5172022-02-22