Winding Rules 缠绕规则

在实现一个镂空的效果时,发下路径的方向,会影响最终实现的效果,所以进一步研究了一下。

填充路径所包含的区域时,会通过缠绕规则来判断需要填充的区域。通过给定区域内的任意一点到路径外画一条射线,根据与路径的交叉数判断点是否在区域内。

缠绕规则:

  • NSNonZeroWindingRule:非零缠绕。射线从左到右每交叉路径一次+1,从右到左每交叉一次-1。如果最终交叉数为0,则该点在路径之外;如果交叉数不为0,则在路径之内。默认缠绕规则。
  • NSEvenOddWindingRule:奇偶缠绕。计算射线与路径的交叉总数,如果为偶数,则在路径之外;如果为奇数,则在路径之内,需要填充。

填充操作适用于开放式路径和闭合路径。开放式路径会从路径的最后一个点到第一个点创建一个隐式的线(不渲染),来闭合路径。

https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/CocoaDrawingGuide/Paths/Paths.html#//apple_ref/doc/uid/TP40003290-CH206-BAJIJJGD 中的 Winding Rules中:
When you fill a partial subpath, NSBezierPath closes it for you automatically by creating an implicit (non-rendered) line from the first to the last point of the subpath.
文档中描述是从第一个点到最后一个点,但是根据分析与文档上的图以及实验,图与结果相同,但是描述错误,下面会详细介绍。如果是我理解错误,恳请指出。

本文demo

闭合路径

1. 非零缠绕:外边框和内边框同一方向

    CGRect aRect = CGRectMake(100, 100, 200, 200);
    UIBezierPath * aPath = [UIBezierPath bezierPathWithRect:aRect];
    CGRect bRect = CGRectInset(aRect, 50, 50);
    UIBezierPath * bPath = [UIBezierPath bezierPathWithRect:bRect];
    
    [aPath appendPath:bPath];
    
    CAShapeLayer * shapeLayer = [CAShapeLayer layer];
    shapeLayer.path = aPath.CGPath;
    shapeLayer.fillColor = [UIColor yellowColor].CGColor;
    
    [self.view.layer addSublayer:shapeLayer];
winding_rules_1.png

内部的点向外画射线,由于两个贝塞尔曲线是同向,射线由右至左跨过路径两次,aRect 以内的所有的点的射线交叉数只有两种情况:0-1=-1,或者0-1-1=-2。都不为0,所以内部的点都在路径之内,需要渲染。

2. 非零缠绕:内边框与外边框反向

// - (UIBezierPath *)bezierPathByReversingPath;  将路径翻转。
// 上面代码只需要修改 bPath
UIBezierPath * bPath = [[UIBezierPath bezierPathWithRect:bRect] bezierPathByReversingPath];
winding_rules_2.png

分为两种情况:

  • bPath 以内的点的射线与路径交叉只有一种:0+1-1=0,因此 bPath 内部的点都在路径最终路径之外,bPath 以内的点不需要渲染。
  • bPath 以外 aPath 以内的点的射线与路径交叉有两种:0-1=-1,或者0+1+1-1=1。两种情况都不为0,所以在路径之内,需要渲染。

奇偶缠绕规则

奇偶缠绕规则下,与内外路径方向无影响。默认 fillRule 为非零,添加如下代码。

shapeLayer.fillRule = kCAFillRuleEvenOdd;

只判断射线与路径的交叉,所以有两种情况:

  • bRect内部的点,射线与路径的交叉有两个,为偶数,所以不在范围内,不需要渲染。
  • bRect之外aRect以内,射线与路径的交叉数可能为1或3,为奇数,所以需要渲染。

开放式路径

盗取苹果文档的图来分析一下,图 c、d 中的射线,按照从上到下从 左->右 命名为 p1,p2,p3。

winding_rules_3.png

非零缠绕规则下 图c:

  • p1 从 左->右 穿过隐式路径,在从 左->右 穿过路径,,0+1+1=2,在路径之内,需要渲染。
  • p2 先是从 左->右 穿过路径,再从 右->左 穿过,0+1-1=0,不在路径之内,不需要渲染。
  • p3 两次从 左->右 穿过路径,0+1+1=2,在路径之内,需要渲染。

这里分析一下隐式线的方向问题,修改一下 p2 的方向为垂直向上。

  • 隐式线的方向是 start->end,首先从 右->左 穿过路径,然后还是从 右->左 穿过隐式线,0-1-1=-2,不为0,应该是属于路径之内,需要渲染的,与原结果冲突。
  • 隐式线的方向是 end->start,首先从 右->左 穿过路径,然后从 左->右 穿过隐式线,0-1+1=0,为0不在路径内,不需要渲染,与原结果相同。

奇偶缠绕规则下 图d

  • p1 穿过隐式线和一次路径,共两次,偶数,不在范围内,不会渲染。
  • p2和p3 穿过两次路径,共两次,偶数,不在范围内,不会渲染。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容