多边形的平行线填充算法

最近做的一个小算法,使用平行线填充一个多边形区域。用过 AutoCAD 的同学应该知道,可以选定一个区域,指定平行线的角度 α 和平行线之间的间距 spacing 就可以填充区域了。

image

图 1 AutoCAD 中的平行线填充效果

这里做的平行线填充实质就是如何计算多边形与各个平行直线的交点,最后将相关的交点连接之后就可以了。具体做的过程如下:

对于一个多边形区域,我们定义这个类为 Boundary,一个 Boundary 里面可以有多个 polygon,其中一个 outer polygon 和多个 inner polygon

  1. 给定的函数借口如果是 List<List<Line>> HatchParallel(Point origin, float angle, float spacing); 其中 origin 是参考点,假设平行线填充无线扩展,那其中的一条直线一定会通过这个参考点,angle 指直线的方向,spacing 指直线之间的距离;返回值的 List<Line> 为一条平行线与 boundary 相交时出现的一系列线段,List<List<Line>> 指一系列平行线与 boundary 相交出现的系列线段;

  2. 根据 angle 值,以 origin 为中心,顺时针旋转直线和 boundary,使平行线变成水平,见图 2;

image

图 2 顺时针旋转图形

  1. 计算旋转后的 boundary 的各个顶点的 Y 值最大值和最小值,由此得到从下到上各个平行线的 Y 值范围

    minY≤Y≤maxY
    
  2. 对于其中的一条平行线,我们定义一个函数 List<Line> ChipLine(float y, Boundary boudary) 对于一个 boundary 可以看成 List<Polygon> polygonList。这里首先计算每个 polygon 和水平线之间的交点情况,接口为 List<Point> ChipLine(float y, Polygon polygon)

    • 4.1 水平线和一个 polygon,其交线的一个情况可能是图 3 所示,其交点分别为 1~7。求交点的过程可以仅用多边形各个点的 Y 值与 origin 的 Y 值进行比较就得到了。相邻的两个点之间的距离可以构成一条线段,但是并不是每条线段都是需要保留的,比如 [5, 6] 就是一定要删去的,而 [3, 4] 可能需要保留也可能需要删除,其余的都是一定要保留的。

    • 4.2 对于 [3, 4],假设我们是需要显示的,如图 4,那么对于 outer polygon,我们保留 3, 4 点,而对于 inner polygon,我们不保留 3, 4 点(这个后面会有解释)。

    • 4.3 这里就先假设我们需要显示 [3, 4],并且多边形是 outer polygon,于是对于 List<Point> ChipLine(float y, Polygon polygon) 函数的返回结果为点列 {1, 2, 2, 3, 3, 4, 4, 5, 6, 7}。可以看到返回的是每条需要保留的线段的端点。

      image

      图 3 水平线与多边形的相交情况

      image

      图 4 显示与多边形边界重合的平行填充线

    • 4.4 如何判断两个点是否需要保留,可以取两个相邻端点的中点,并判断中点与多边形的内外关系:

A、如果中点在多边形内部,则这2点连接的线段就一定在多边形内部,于是这2点就一定要保留;

B、如果中点在多边形外部,则这2点连接的线段就一定在多边形外部,则这2点就一定要删除;

C、如果中点在多边形的边上,则根据条件判断这2个点是不是需要保留,判断规则如下:

C1、如果我们需要显示与多边形的边重合的平行填充线

C11、当前多边形是outer多边形,则保留这2个点;

C12、当前多边形是inner多边形,则删除这2个点;

C2、如果我们不需要显示与多边形的边重合的平行填充线

C21、当前多边形是outer多边形,则删除这2个点;

C22、当前多边形是inner多边形,则保留这2个点;

  1. 对一个 boundary 中的全部多边形进行第 4 步骤后,将全部的点保留在一个 List<Point> 当中,随后对全部的点按照 X 轴进行排序,得到经过排序的点列 {p0, p1, p2, p3, …, p2n-1}。由第 4 步可知,得到的点列的数量一定是 2 的倍数。

  2. 将点列的相邻两个点提取出来,{p0, p1}、{ p2, p3}、…、{ p2n-2, p2n-1},并由此构成一个线段的列 List<Line> 并返回,由此完成了 List<Line> ChipLine(float y, Boundary boudary)

    至于在第 4 步骤当中,为什么对 outer 多边形和 inner 多边形的边界边两点的保留还是删除采用不同的方式,可以见图 5。假设保留了 inner 多边形与水平线的交点,则第 5 步得到的点列为 {1, 2, 3, 4, 4, 5, 5, 6, 6, 7, 8, 9},由此后面两个配对得到 List<Line> 的时候就没有线段 [2, 3] 了,所以应当对 inner polygonouter polygon 采用不同的处理方式。

    image

    图 5 水平线与一个 boundary 的相交情况

贴一些自己做的效果图吧

image

图6

image

图 7 保留与边界重合的平行线

image

图 8 不保留与边界重合的平行线

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

推荐阅读更多精彩内容