我设计了这个代码来生成一个Bezier Path,该路径用作CAShapeLayer屏蔽UIView的路径(视图的高度和宽度是可变的)

这个代码生成一个具有尖锐边缘的三角形,但我想让它圆角!我花了两个小时尝试与addArcWithCenter …,lineCapStyle和lineJoinStyle等工作,但似乎没有任何工作.

UIBezierPath *bezierPath = [UIBezierPath bezierPath];

CGPoint center = CGPointMake(rect.size.width / 2,0);
CGPoint bottomLeft = CGPointMake(10,rect.size.height - 0);
CGPoint bottomright = CGPointMake(rect.size.width - 0,rect.size.height - 0);

[bezierPath movetoPoint:center];
[bezierPath addLinetoPoint:bottomLeft];
[bezierPath addLinetoPoint:bottomright];
[bezierPath closePath];

所以我的问题是,我将如何舍弃UIBezierPath中的三角形的所有边缘(我需要子层,多个路径等)?

N.B我没有绘制这个BezierPath,所以在drawRect中的所有CGContext …函数都不能帮助我:(

谢谢!

解决方法

编辑

FWIW:这个答案通过解释CGPathAddArcToPoint(…)为您做什么来达到教育目的.我强烈建议您阅读它,因为它将帮助您了解和欣赏CGPath API.然后,您应该继续使用,如an0’s answer所示,而不是在您的应用程序中边缘边缘时使用此代码.这个代码只能用作参考,如果你想玩,并学习这样的几何计算.

原来的答案

因为我觉得这样的问题很有趣,我不得不回答:)

这是一个很长的答案.没有简短版本:D

注意:为了我自己的简单,我的解决方案正在对用于形成三角形的点进行一些假设,例如:

>三角形的面积足够大以适合圆角(例如,三角形的高度大于角落中圆圈的直径),我没有检查或试图阻止任何类型的奇怪的结果,否则发生.
>角落按逆时针顺序排列.您可以使其适用于任何订单,但是为了简单起见,它感觉到一个足够的限制.

如果你想要的话,你可以使用相同的技术来舍入任何多边形,只要它是非常凸的(即不是一个尖锐的星).我不会解释如何做,但它遵循相同的原则.

这一切都是以三角形开始的,你想绕一些半径的角落r:

圆角三角形应该包含在尖角三角形中,所以第一步是找到尽可能靠近角落的位置,在那里您可以配合半径为r的圆.

这样做的一个简单方法是创建平行于三角形中三个边的3条新线,并将距离r的每一个向内移动,与原始边正交.

为此,您可以计算每条线的斜率/角度和要应用于两个新点的偏移量:

CGFloat angle = atan2f(end.y - start.y,end.x - start.x);

CGVector offset = CGVectorMake(-sinf(angle)*radius,cosf(angle)*radius);

注意:为了清楚起见,我使用的是CGVector类型(在iOS 7中可用),但是您也可以使用一个或多个大小来处理以前的操作系统版本.

那么您将偏移量添加到每一行的起点和终点:

CGPoint offsetStart = CGPointMake(start.x + offset.dx,start.y + offset.dy);

CGPoint offsetEnd   = CGPointMake(end.x + offset.dx,end.y + offset.dy);

当你做,你会看到三条线在三个地方相交:

每个交点正好是两侧的距离r(假设三角形足够大,如上所述).

您可以计算两行的交点为:

//       (x1⋅y2-y1⋅x2)(x3-x4) - (x1-x2)(x3⋅y4-y3⋅x4)
// px =  –––––––––––––––––––––––––––––––––––––––––––
//            (x1-x2)(y3-y4) - (y1-y2)(x3-x4)

//       (x1⋅y2-y1⋅x2)(y3-y4) - (y1-y2)(x3⋅y4-y3⋅x4)
// py =  –––––––––––––––––––––––––––––––––––––––––––
//            (x1-x2)(y3-y4) - (y1-y2)(x3-x4)

CGFloat intersectionX = ((x1*y2-y1*x2)*(x3-x4) - (x1-x2)*(x3*y4-y3*x4)) / ((x1-x2)*(y3-y4) - (y1-y2)*(x3-x4));
CGFloat intersectionY = ((x1*y2-y1*x2)*(y3-y4) - (y1-y2)*(x3*y4-y3*x4)) / ((x1-x2)*(y3-y4) - (y1-y2)*(x3-x4));

CGPoint intersection = CGPointMake(intersectionX,intersectionY);

其中(x1,y1)至(x2,y2)为第一行,(x3,y3)至(x4,y4)为第二行.

如果然后在每个交点上放置一个半径为r的圆圈,您可以看到它将确实为圆角三角形(忽略三角形和圆圈的不同线宽):

现在要创建一个圆形的三角形,你想要创建一个从原来的三角形正交于交点的点到一条线到一条线(等)的路径.这也是圆形与原来三角形相切的点.

知道三角形中所有三边的斜率,角半径和圆心(交点),每个圆角的起始和终止角是该边的斜率 – 90度.要将这些东西分组在一起,我在代码中创建了一个结构体,但如果不想要,则不需要:

typedef struct {
    CGPoint centerPoint;
    CGFloat startAngle;
    CGFloat endAngle;
} CornerPoint;

为了减少代码重复,我创建了一个自己计算交点的方法,以及从一个点到另一个点给出的一个点的角度到最后一点(它不是闭合的,因此它不是一个三角形):

代码如下(这只是我上面显示的代码,放在一起):

- (CornerPoint)roundedCornerWithLinesFrom:(CGPoint)from
                                      via:(CGPoint)via
                                       to:(CGPoint)to
                               withRadius:(CGFloat)radius
{
    CGFloat fromAngle = atan2f(via.y - from.y,via.x - from.x);
    CGFloat toAngle   = atan2f(to.y  - via.y,to.x  - via.x);

    CGVector fromOffset = CGVectorMake(-sinf(fromAngle)*radius,cosf(fromAngle)*radius);
    CGVector toOffset   = CGVectorMake(-sinf(toAngle)*radius,cosf(toAngle)*radius);


    CGFloat x1 = from.x +fromOffset.dx;
    CGFloat y1 = from.y +fromOffset.dy;

    CGFloat x2 = via.x  +fromOffset.dx;
    CGFloat y2 = via.y  +fromOffset.dy;

    CGFloat x3 = via.x  +toOffset.dx;
    CGFloat y3 = via.y  +toOffset.dy;

    CGFloat x4 = to.x   +toOffset.dx;
    CGFloat y4 = to.y   +toOffset.dy;

    CGFloat intersectionX = ((x1*y2-y1*x2)*(x3-x4) - (x1-x2)*(x3*y4-y3*x4)) / ((x1-x2)*(y3-y4) - (y1-y2)*(x3-x4));
    CGFloat intersectionY = ((x1*y2-y1*x2)*(y3-y4) - (y1-y2)*(x3*y4-y3*x4)) / ((x1-x2)*(y3-y4) - (y1-y2)*(x3-x4));

    CGPoint intersection = CGPointMake(intersectionX,intersectionY);

    CornerPoint corner;
    corner.centerPoint = intersection;
    corner.startAngle  = fromAngle - M_PI_2;
    corner.endAngle    = toAngle   - M_PI_2;

    return corner;
}

然后我使用该代码3次来计算3个角:

CornerPoint leftCorner  = [self roundedCornerWithLinesFrom:right
                                                       via:left
                                                        to:top
                                                withRadius:radius];

CornerPoint topCorner   = [self roundedCornerWithLinesFrom:left
                                                       via:top
                                                        to:right
                                                withRadius:radius];

CornerPoint rightCorner = [self roundedCornerWithLinesFrom:top
                                                       via:right
                                                        to:left
                                                withRadius:radius];

现在,拥有所有必要的数据,启动我们创建实际路径的部分.我将依赖于CGPathAddArc将从当前点到起点添加一条直线,而不必自己绘制这些行(这是记录的行为).

我手动必须计算的唯一点是路径的起点.我选择右下角的开始(没有具体原因).从那里,您只需在起点和终点角度的交点中加上中心的弧:

CGMutablePathRef roundedTrianglePath = CGPathCreateMutable();
// manually calculated start point
CGPathMovetoPoint(roundedTrianglePath,NULL,leftCorner.centerPoint.x + radius*cosf(leftCorner.startAngle),leftCorner.centerPoint.y + radius*sinf(leftCorner.startAngle));
// add 3 arcs in the 3 corners 
CGPathAddArc(roundedTrianglePath,leftCorner.centerPoint.x,leftCorner.centerPoint.y,radius,leftCorner.startAngle,leftCorner.endAngle,NO);
CGPathAddArc(roundedTrianglePath,topCorner.centerPoint.x,topCorner.centerPoint.y,topCorner.startAngle,topCorner.endAngle,rightCorner.centerPoint.x,rightCorner.centerPoint.y,rightCorner.startAngle,rightCorner.endAngle,NO);
// close the path
CGPathCloseSubpath(roundedTrianglePath);

看起来像这样:

最终结果没有所有的支持线,看起来像这样:

ios – UIBezierPath具有圆形边缘的三角形的更多相关文章

  1. ios – 如何使用渐变填充UIBezierPath?

    我使用UIBezierPath绘制了一个图形.我可以使用纯色填充图表下方的区域,但我想用渐变而不是纯色填充图表下方的区域.但我不知道如何使渐变仅适用于图形而不是整个视图,我已经阅读了一些问题,但没有找到任何适用的问题.这是主要的图形绘制代码:我一直试图通过试验下面的代码来填充图表,但说实话,不是吗?

  2. ios – 使用watchOS 2在Apple Watch上渲染折线图

    我正在尝试使用watchOS2在AppleWatch上渲染线条/步骤图.与iOS9不同,watchOS2不支持Quartz.它只支持CoreGraphics.我尝试编写一些代码来绘制折线图但我得到一个错误“CGContextRestoreGState:无效的上下文0x0.这是一个严重的错误.这个应用程序,或它使用的库,正在使用无效的上下文,从而有助于整体系统稳定性和可靠性降低.这个通知是礼貌的:请

  3. ios – 在两个圆形UIBezierPaths之间进行动画制作时的奇怪行为

    问题我正在创造爆炸环效果.我正在使用多个CAShapeLayer并为每个层创建一个UIBezierPath.初始化视图时,所有图层都有一个宽度为0的路径.触发动作时,环会为更大的路径设置动画.如下面的演示所示,每个图层的右边缘比每个圆形图层的其余部分的动画效果要慢.演示码绘制图层:更新图层解决方法避免从路径宽度为0开始.从0开始缩放并且微小的浮点误差放大是非常困难的.非常接近零的值在浮点上不是很连续(嗯……

  4. ios – 如何使用渐变颜色填充贝塞尔曲线路径

    我的自定义UIView绘图(_rect:CGRect)函数中有一个UIBezierPath.我想用渐变色填充路径.请任何人都可以指导我如何做到这一点.我需要用渐变颜色填充剪辑,然后用黑色描边路径.SO中有一些帖子无法解决问题.例如Swift:Gradientalongabezierpath(usingCALayers)这篇文章指导如何在UIView中绘制图层,而不是在UIBezierPath中绘制

  5. ios – 如何为UIBezierPath提供cornerRadius

    我使用以下代码创建了一个矩形,现在我需要舍入这个矩形的角.但是我找不到一个名为layer.cornerRadius的属性,有人可以帮帮我吗?如果你想要特定的角落在下面的方法使用.

  6. ios – UIBezierPath具有圆形边缘的三角形

    我设计了这个代码来生成一个BezierPath,该路径用作CAShapeLayer屏蔽UIView的路径这个代码生成一个具有尖锐边缘的三角形,但我想让它圆角!,lineCapStyle和lineJoinStyle等工作,但似乎没有任何工作.所以我的问题是,我将如何舍弃UIBezierPath中的三角形的所有边缘?N.B我没有绘制这个BezierPath,所以在drawRect中的所有CGContext…

  7. ios – 如何在UIView中绘制不同颜色的多个UIBezierPath

    我想在uiview中绘制多个UIBezierPath,具有不同的笔画和填充颜色.这是代码问题是笔画和填充颜色是在当前上下文中设置的.在同一个CGContextRef中可以用不同的颜色绘制不同的UIBezierPath吗?或者我必须在单独的uiview中绘制每个UIBezierPath?解决方法你应该添加表明这些是在这种情况下必须进一步使用的新颜色.你每次打电话之前都应该这样做以便用这些颜色绘制路径.不需要为每个贝塞尔路径使用新的视图.

  8. iOS UIImage剪辑路径

    我正在使用用户使用UIBezierPath选择其中一部分的图像.如何删除/清除/清除不是该选择的一部分的所有内容?解决方法有一条路很容易.只需将路径设置为剪切路径:如果要使用多个路径的并集,则更难,因为Quartz没有任何直接计算两个路径的并集的函数.一种方法是将每个路径逐个填入蒙版,然后通过掩码绘制图像:

  9. ios – 完全删除UIBezierPath和CAShapeLayer绘制的项目

    在纵向模式下(第一次绘图)旋转到横向后,为什么旧的形状:我正在使用UIBezierPath&CAShapeLayer绘制圆圈&自定义UIView图层中的线条.问题在于能够吸引新的圈子和新的圈子.设备旋转后成功的线条,我无法删除旧的绘制形状.在设备旋转后绘制的新形状是完美的,我只需要从屏幕上移除那些粘在屏幕上的旧形状.附图像.解决方法您可以删除以前的CAShapeLayer(使用removeFrom

  10. ios – 如何在Swift中更改UIBezierPath的颜色?

    我有一个UIBezierPath的实例,我想将笔画的颜色改为黑色以外的东西.有谁知道如何在斯威夫特这样做?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部