美文网首页
【iOS】使用一组点画出平滑的曲线

【iOS】使用一组点画出平滑的曲线

作者: haifengmay | 来源:发表于2016-11-21 19:40 被阅读2956次

今天在项目中需要人脸上的点来勾勒出人脸的轮廓,我的想法是将要画的点存入一个数组,使用了UIBezierPath来连接每一个点。但是这样画出来的图是折线,显得过于生硬。查了若干资料,后来终于在stackoverflow上找到了一个很好的解决办法

这个做法的原理其实是在每两个点之间加入一些点,来使得两个点可以平滑的过度。这其实就是Centripetal Catmull–Rom spline的思想。

于是便照葫芦画瓢,写出了我的代码

#define POINT(_INDEX_) [(NSValue *)[points objectAtIndex:_INDEX_] CGPointValue]
- (void)smoothedPathWithPoints:(NSArray *) pointsArray andGranularity:(NSInteger)granularity {
    
    NSMutableArray *points = [pointsArray mutableCopy];
    
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGContextSetAllowsAntialiasing(context, YES);
    CGContextSetStrokeColorWithColor(context, [UIColor greenColor].CGColor);
    CGContextSetLineWidth(context, 0.6);
    
    UIBezierPath *smoothedPath = [UIBezierPath bezierPath];
    
    // Add control points to make the math make sense
    [points insertObject:[points objectAtIndex:0] atIndex:0];
    [points addObject:[points lastObject]];
    [smoothedPath moveToPoint:POINT(0)];
    
    for (NSUInteger index = 1; index < points.count - 2; index++) {
        CGPoint p0 = POINT(index - 1);
        CGPoint p1 = POINT(index);
        CGPoint p2 = POINT(index + 1);
        CGPoint p3 = POINT(index + 2);
        
        // now add n points starting at p1 + dx/dy up until p2 using Catmull-Rom splines
        for (int i = 1; i < granularity; i++) {
            
            float t = (float) i * (1.0f / (float) granularity);
            float tt = t * t;
            float ttt = tt * t;
            
            CGPoint pi; // intermediate point
            pi.x = 0.5 * (2*p1.x+(p2.x-p0.x)*t + (2*p0.x-5*p1.x+4*p2.x-p3.x)*tt + (3*p1.x-p0.x-3*p2.x+p3.x)*ttt);
            pi.y = 0.5 * (2*p1.y+(p2.y-p0.y)*t + (2*p0.y-5*p1.y+4*p2.y-p3.y)*tt + (3*p1.y-p0.y-3*p2.y+p3.y)*ttt);
            [smoothedPath addLineToPoint:pi];
        }
        
        // Now add p2
        [smoothedPath addLineToPoint:p2];
    }
    
    // finish by adding the last point
    [smoothedPath addLineToPoint:POINT(points.count - 1)];
    
    CGContextAddPath(context, smoothedPath.CGPath);
    CGContextDrawPath(context, kCGPathStroke);
    
}

相关文章

  • 【iOS】使用一组点画出平滑的曲线

    今天在项目中需要人脸上的点来勾勒出人脸的轮廓,我的想法是将要画的点存入一个数组,使用了UIBezierPath来连...

  • QCustomPlot之平滑曲线下(九)

    接上篇QCustomPlot之平滑曲线上(八),上篇只是实现了平滑曲线的绘制,但是并没有实现平滑曲线与0点线之间的...

  • 贝塞尔曲线原理及在iOS中使用介绍

    贝塞尔曲线是指可以通过一些控制点去控制曲线的形状并且保持曲线的平滑特性,不会让人感觉到突兀。在iOS开发中,贝塞尔...

  • Echarts 折线图(line) 平滑曲线单调性问题

    问题 最近在使用echarts画折线图时,遇到一个奇怪的问题。在双曲线的line图中,曲线设置为平滑时,相邻两点的...

  • iOS开发之连续平滑曲线图

    前言 连续平滑曲线的需求,在项目中经常会遇到,有很多种方式可以画出来,网上也有很多教程,但是总是不能画出自己满意的...

  • 平滑的曲线

    我不能一步登天,只能走着平滑的曲线。一是我的实力不够,一是我比较谨小慎微。小时候我感到天很近,我的雄心壮志可以升到...

  • [前端]使用曲线将多点连成一条平滑的曲线

    之前在写一个项目需要把多点连成平滑的曲线,而且这些点是无法预知的。开始想到用贝塞尔曲线,但是具体贝塞尔曲线的控制点...

  • 曲线调色经验

    用曲线调色有诸多好处 1.曲线可以得到过度更平滑和幅度更大的颜色变化 该图片用蓝色曲线实现了平滑的颜色过渡,使得蓝...

  • 我的图形

    我喜欢平滑的曲线 以及 由平滑曲线连接而成的图形 如果这个图形 刚好是由各种绿色填充而成 那么它将属于我

  • SU曲线插件Curvizard2.4a中文破解版免费下载附安装教

    Curvizard是一款曲线编辑工具,包含焊接曲线、炸开曲线、清理曲线、简化曲线、平滑曲线、转为虚线等曲线编辑功能...

网友评论

      本文标题:【iOS】使用一组点画出平滑的曲线

      本文链接:https://www.haomeiwen.com/subject/bzwupttx.html