-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathutils.py
More file actions
548 lines (428 loc) · 17.9 KB
/
utils.py
File metadata and controls
548 lines (428 loc) · 17.9 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
# -*- coding:utf-8 -*-
# ==============================================================================
# Imports
# ==============================================================================
import functools
import collections
from operator import attrgetter
from typing import List, Tuple, Any, Optional, Union
# Rhino Libraries
import Rhino
import Rhino.Geometry as geo
import ghpythonlib.components as ghcomp
# ==============================================================================
# Constants
# ==============================================================================
BIGNUM = 10000000
ROUNDING_PRECISION = 6 # 반올림 소수점 자리수
# Tolerances
TOL = 0.01 # 기본 허용 오차
DIST_TOL = 0.01
AREA_TOL = 0.1
OP_TOL = 0.00001
CLIPPER_TOL = 0.0000000001
# ==============================================================================
# Decorators
# ==============================================================================
def convert_io_to_list(func):
"""입력과 출력을 리스트 형태로 일관되게 만들어주는 데코레이터"""
@functools.wraps(func)
def wrapper(*args, **kwargs):
new_args = []
for arg in args:
if isinstance(arg, geo.Curve):
arg = [arg]
new_args.append(arg)
result = func(*new_args, **kwargs)
if isinstance(result, geo.Curve):
result = [result]
if hasattr(result, "__dict__"):
for key, values in result.__dict__.items():
if isinstance(values, geo.Curve):
setattr(result, key, [values])
return result
return wrapper
# ==============================================================================
# Core Geometry Utilities
# ==============================================================================
def get_distance_between_points(point_a: geo.Point3d, point_b: geo.Point3d) -> float:
"""두 점 사이의 거리를 계산합니다."""
return round(point_a.DistanceTo(point_b), ROUNDING_PRECISION)
def get_distance_between_point_and_curve(point: geo.Point3d, curve: geo.Curve) -> float:
"""점과 커브 사이의 최단 거리를 계산합니다."""
_, param = curve.ClosestPoint(point)
dist = point.DistanceTo(curve.PointAt(param))
return round(dist, ROUNDING_PRECISION)
def get_distance_between_curves(curve_a: geo.Curve, curve_b: geo.Curve) -> float:
"""두 커브 사이의 최소 거리를 계산합니다."""
_, pt_a, pt_b = curve_a.ClosestPoints(curve_b)
dist = pt_a.DistanceTo(pt_b)
return round(dist, ROUNDING_PRECISION)
def has_intersection(
curve_a: geo.Curve,
curve_b: geo.Curve,
plane: geo.Plane = geo.Plane.WorldXY,
tol: float = TOL,
) -> bool:
"""두 커브가 교차하는지 여부를 확인합니다."""
return geo.Curve.PlanarCurveCollision(curve_a, curve_b, plane, tol)
def get_pts_from_crv_crv(
curve_a: geo.Curve, curve_b: geo.Curve, tol: float = TOL
) -> List[geo.Point3d]:
"""두 커브 사이의 교차점을 계산합니다."""
intersections = geo.Intersect.Intersection.CurveCurve(curve_a, curve_b, tol, tol)
if not intersections:
return []
return [event.PointA for event in intersections]
def get_pts_from_crvs(crvs: List[geo.Curve], tol=TOL) -> List[geo.Point3d]:
intersection = ghcomp.MultipleCurves(crvs)
return list(geo.Point3d.CullDuplicates(list(intersection.points), tol))
def explode_curve(curve: Union[geo.Curve, List[geo.Curve]]) -> List[geo.Curve]:
"""커브를 분할하여 개별 세그먼트(직선) 리스트로 반환합니다."""
if not curve:
return []
# PolyCurve인 경우, 내부 세그먼트들을 직접 반환
if isinstance(curve, geo.PolyCurve):
return list(curve.DuplicateSegments())
# 일반 커브는 Span 기준으로 분할
segments = []
for i in range(curve.SpanCount):
param_start, param_end = curve.SpanDomain(i)
pt_start = curve.PointAt(param_start)
pt_end = curve.PointAt(param_end)
segments.append(geo.LineCurve(pt_start, pt_end))
return segments
def get_outside_perp_vec_from_pt(pt: geo.Point3d, region: geo.Curve) -> geo.Vector3d:
_, param = region.ClosestPoint(pt)
vec_perp_outer = region.PerpendicularFrameAt(param)[1].XAxis
if region.ClosedCurveOrientation() != geo.CurveOrientation.Clockwise:
vec_perp_outer = -vec_perp_outer
return vec_perp_outer
def get_pts_by_length(
crv: geo.Curve, length: float, include_start: bool = False
) -> List[geo.Point3d]:
"""커브를 주어진 길이로 나누는 점을 구한다."""
params = crv.DivideByLength(length, include_start)
# crv가 length보다 짧은 경우
if not params:
return []
return [crv.PointAt(param) for param in params]
def get_vector_from_pts(pt_a: geo.Point3d, pt_b: geo.Point3d) -> geo.Vector3d:
"""두 점 사이의 벡터를 계산합니다."""
return geo.Vector3d(pt_b.X - pt_a.X, pt_b.Y - pt_a.Y, pt_b.Z - pt_a.Z)
def get_vertices(curve: geo.Curve) -> List[geo.Point3d]:
"""커브의 모든 정점(Vertex)들을 추출합니다."""
vertices = [curve.PointAt(curve.SpanDomain(i)[0]) for i in range(curve.SpanCount)]
if not curve.IsClosed:
vertices.append(curve.PointAtEnd)
return vertices
def move_curve(curve: geo.Curve, vector: geo.Vector3d) -> geo.Curve:
"""커브를 주어진 벡터만큼 이동시킨 복사본을 반환합니다."""
moved_curve = curve.Duplicate()
moved_curve.Translate(vector)
return moved_curve
def move_brep(brep: geo.Brep, vector: geo.Vector3d) -> geo.Brep:
"""Brep를 주어진 벡터만큼 이동시킨 복사본을 반환합니다."""
moved_brep = brep.Duplicate()
moved_brep.Translate(vector)
return moved_brep
def split_curve_at_pts(
curve: geo.Curve, points: List[geo.Point3d], tol: float = TOL
) -> List[geo.Curve]:
"""커브를 주어진 점들에서 분할합니다."""
if len(points) == 0:
return [curve]
params = []
for pt in points:
ok, param = curve.ClosestPoint(pt, tol)
if ok:
params.append(param)
if not params or len(params) == 0:
return [curve]
params = list(set(params)) # 중복 제거
params.sort()
split_curves = ghcomp.Shatter(curve, params)
if isinstance(split_curves, geo.Curve):
return [split_curves]
return split_curves if split_curves else [curve]
def get_inside_check_pt(crv):
"""crv 내부의 임의점"""
_, ply = crv.TryGetPolyline()
mesh = geo.Mesh.CreateFromClosedPolyline(ply)
return mesh.Faces.GetFaceCenter(0)
def is_pt_inside(pt: geo.Point3d, crv: geo.Curve, tol: float = TOL) -> bool:
# -1: 일치, 0: 밖, 1: 안
result = ghcomp.ClipperComponents.PolylineContainment(
crv, pt, geo.Plane.WorldXY, tol
)
return result == 1
def move_curve_endpoint(
curve: geo.Curve, target: geo.Point3d, which: str = "start"
) -> geo.Curve:
"""커브의 시작점 또는 끝점을 주어진 좌표로 이동시킵니다."""
pts = get_vertices(curve)
if which not in ("start", "end"):
raise ValueError("which는 'start' 또는 'end'만 허용됩니다.")
if which == "start":
pts[0] = target
else:
pts[-1] = target
return geo.PolylineCurve(pts)
def is_pt_on_crv(pt: geo.Point3d, crv: geo.Curve, tol: float = TOL) -> bool:
"""점이 커브 위에 있는지 확인합니다."""
if not pt or not crv:
return False
_, param = crv.ClosestPoint(pt, tol)
if param is None:
return False
closest_pt = crv.PointAt(param)
return pt.DistanceTo(closest_pt) <= tol
# ==============================================================================
# Advanced Curve & Region Operations
# ==============================================================================
def get_overlapped_curves(curve_a: geo.Curve, curve_b: geo.Curve) -> List[geo.Curve]:
"""두 커브가 겹치는 구간의 커브들을 반환합니다."""
if not has_intersection(curve_a, curve_b):
return []
intersection_points = get_pts_from_crv_crv(curve_a, curve_b)
explode_points = ghcomp.Explode(curve_a, True).vertices + intersection_points
if not explode_points:
return []
params = [ghcomp.CurveClosestPoint(pt, curve_a).parameter for pt in explode_points]
segments = ghcomp.Shatter(curve_a, params)
overlapped_segments = [seg for seg in segments if has_intersection(seg, curve_b)]
if not overlapped_segments:
return []
return geo.Curve.JoinCurves(overlapped_segments)
def get_overlapped_length(curve_a: geo.Curve, curve_b: geo.Curve) -> float:
"""두 커브가 겹치는 총 길이를 계산합니다."""
overlapped_curves = get_overlapped_curves(curve_a, curve_b)
if not overlapped_curves:
return 0.0
return sum(crv.GetLength() for crv in overlapped_curves)
def has_region_intersection(
region_a: geo.Curve, region_b: geo.Curve, tol: float = TOL
) -> bool:
"""두 닫힌 영역 커브가 교차(겹침 포함)하는지 확인합니다."""
relationship = geo.Curve.PlanarClosedCurveRelationship(
region_a, region_b, geo.Plane.WorldXY, tol
)
return relationship != geo.RegionContainment.Disjoint
def is_region_inside_region(
region: geo.Curve, other_region: geo.Curve, tol: float = TOL
) -> bool:
"""'region'이 'other_region' 내부에 완전히 포함되는지 확인합니다."""
relationship = geo.Curve.PlanarClosedCurveRelationship(
region, other_region, geo.Plane.WorldXY, tol
)
return relationship == geo.RegionContainment.AInsideB
def get_outline_from_closed_brep(brep: geo.Brep, plane: geo.Plane) -> geo.Curve:
"""
닫힌 폴리서페이스(Brep)를 받아, 주어진 Plane 기준으로 Contour를 생성하고,
결과 커브들 중 Z값이 가장 낮은 커브를 반환합니다.
brep가 닫힌 Brep가 아니면 TypeError를 발생시킵니다.
"""
if not isinstance(brep, geo.Brep) or not brep.IsSolid:
raise TypeError("입력은 닫힌 Brep(폴리서페이스)만 허용됩니다.")
bbox = brep.GetBoundingBox(True)
contour_start = geo.Point3d(0, 0, bbox.Min.Z)
contour_end = geo.Point3d(0, 0, bbox.Max.Z)
curves = geo.Brep.CreateContourCurves(
brep, contour_start, contour_end, (bbox.Max.Z - bbox.Min.Z)
)
if not curves or len(curves) == 0:
return None
# Z값이 가장 낮은 커브 선택 (평균 Z값 기준)
def avg_z(curve):
return curve.PointAtStart.Z
return min(curves, key=avg_z)
def offset_regions_inward(
regions: Union[geo.Curve, List[geo.Curve]], dist: float, miter: int = BIGNUM
) -> List[geo.Curve]:
"""영역 커브를 안쪽으로 offset 한다.
단일커브나 커브리스트 관계없이 커브 리스트로 리턴한다.
Args:
region: offset할 대상 커브
dist: offset할 거리
Returns:
offset 후 커브
"""
if not dist:
return regions
return Offset().polyline_offset(regions, dist, miter).holes
def offset_regions_outward(
regions: Union[geo.Curve, List[geo.Curve]], dist: float, miter: int = BIGNUM
) -> List[geo.Curve]:
"""영역 커브를 바깥쪽으로 offset 한다.
단일커브나 커브리스트 관계없이 커브 리스트로 리턴한다.
Args:
region: offset할 대상 커브
dist: offset할 거리
returns:
offset 후 커브
"""
if isinstance(regions, geo.Curve):
regions = [regions]
return [offset_region_outward(region, dist, miter) for region in regions]
def offset_region_outward(
region: geo.Curve, dist: float, miter: float = BIGNUM
) -> geo.Curve:
"""영역 커브를 바깥쪽으로 offset 한다.
단일 커브를 받아서 단일 커브로 리턴한다.
Args:
region: offset할 대상 커브
dist: offset할 거리
Returns:
offset 후 커브
"""
if not dist:
return region
if not isinstance(region, geo.Curve):
raise ValueError("region must be curve")
if not region.IsClosed:
raise ValueError("region must be closed curve")
return Offset().polyline_offset(region, dist, miter).contour[0]
def offset_crv_outward(crv: geo.Curve, dist: float, miter: float = BIGNUM) -> geo.Curve:
"""커브를 바깥쪽으로 offset 한다.
단일 커브를 받아서 단일 커브로 리턴한다.
Args:
crv: offset할 대상 커브
dist: offset할 거리
Returns:
offset 후 커브
"""
if not dist:
return crv
if not isinstance(crv, geo.Curve):
raise ValueError("crv must be curve")
if crv.IsClosed:
raise ValueError("crv must be open curve")
return Offset().polyline_offset(crv, dist, miter).contour
class Offset:
class _PolylineOffsetResult:
def __init__(self):
self.contour: Optional[List[geo.Curve]] = None
self.holes: Optional[List[geo.Curve]] = None
@convert_io_to_list
def polyline_offset(
self,
crvs: List[geo.Curve],
dists: List[float],
miter: int = BIGNUM,
closed_fillet: int = 2,
open_fillet: int = 2,
tol: float = Rhino.RhinoMath.ZeroTolerance,
) -> _PolylineOffsetResult:
"""
Args:
crv (_type_): _description_
dists (_type_): _description_
miter : miter
closed_fillet : 0 = round, 1 = square, 2 = miter
open_fillet : 0 = round, 1 = square, 2 = butt
Returns:
_type_: _PolylineOffsetResult
"""
if not crvs:
raise ValueError("No Curves to offset")
plane = geo.Plane(geo.Point3d(0, 0, crvs[0].PointAtEnd.Z), geo.Vector3d.ZAxis)
result = ghcomp.ClipperComponents.PolylineOffset(
crvs,
dists,
plane,
tol,
closed_fillet,
open_fillet,
miter,
)
polyline_offset_result = Offset._PolylineOffsetResult()
for name in ("contour", "holes"):
setattr(polyline_offset_result, name, result[name])
return polyline_offset_result
class RegionBool:
@convert_io_to_list
def _polyline_boolean(
self, crvs0, crvs1, boolean_type=None, plane=None, tol=CLIPPER_TOL
):
# type: (List[geo.Curve], List[geo.Curve], int, geo.Plane, float) -> List[geo.Curve]
if not crvs0 or not crvs1:
raise ValueError("Check input values")
result = ghcomp.ClipperComponents.PolylineBoolean(
crvs0, crvs1, boolean_type, plane, tol
)
# 결과는 IronPython.Runtime.List (파이썬 list처럼 동작) 이거나 단일 커브일 수 있으므로 통일해서 list로 반환
if not result:
return []
# IronPython.Runtime.List, System.Collections.Generic.List, tuple 등 반복 가능한 결과를 모두 처리
if isinstance(result, geo.Curve):
# 단일 커브 객체
result = [result]
else:
try:
# IEnumerable / IronPython.Runtime.List / tuple / System.Collections.Generic.List 모두 list() 시도로 통일
result = [crv for crv in list(result) if crv]
except TypeError:
# 반복 불가능한 단일 객체인 예외 상황
result = [result]
return result
def polyline_boolean_union(self, crvs0, crvs1, plane=None, tol=CLIPPER_TOL):
# type: (Union[geo.Curve, List[geo.Curve]], Union[geo.Curve, List[geo.Curve]], geo.Plane, float) -> List[geo.Curve]
return self._polyline_boolean(crvs0, crvs1, 1, plane, tol)
def polyline_boolean_difference(self, crvs0, crvs1, plane=None, tol=CLIPPER_TOL):
# type: (Union[geo.Curve, List[geo.Curve]], Union[geo.Curve, List[geo.Curve]], geo.Plane, float) -> List[geo.Curve]
return self._polyline_boolean(crvs0, crvs1, 2, plane, tol)
def get_difference_regions(
regions_a: Union[List[geo.Curve], geo.Curve],
regions_b: Union[List[geo.Curve], geo.Curve],
offset_tol: float = None,
) -> List[geo.Curve]:
"""주어진 두 영역 커브의 차집합을 구합니다.
Args:
regions_a: 차집합의 대상이 되는 영역 커브
regions_b: 차집합에서 제외할 영역 커브
Returns:
차집합 결과 커브들
"""
result = RegionBool().polyline_boolean_difference(regions_a, regions_b)
if offset_tol and result:
result = offset_regions_inward(result, offset_tol)
result = offset_regions_outward(result, offset_tol)
return result
def get_union_regions(regions: List[geo.Curve] = None) -> List[geo.Curve]:
"""주어진 영역 커브들의 합집합을 구합니다.
Args:
regions: 합집합을 구할 영역 커브들
Returns:
합집합 결과 커브들
"""
if not regions:
return []
if len(regions) == 1:
return regions
union_result = list(geo.Curve.CreateBooleanUnion(regions, TOL))
if union_result:
return union_result
union_result = regions[0]
for region in regions[1:]:
union_result = RegionBool().polyline_boolean_union(union_result, region)
if not isinstance(union_result, list):
union_result = [union_result]
return union_result
def get_regions_from_crvs(crvs: List[geo.Curve]) -> List[geo.Curve]:
"""주어진 커브들로부터 닫힌 영역 커브들을 추출합니다.
Args:
crvs: 닫힌 영역을 형성하는 커브들
Returns:
닫힌 영역 커브들
"""
if not crvs:
return []
boolean_regions = geo.Curve.CreateBooleanRegions(
crvs, geo.Plane.WorldXY, False, OP_TOL
) # type: geo.CurveBooleanRegions
regions_count = boolean_regions.RegionCount
result_regions = []
for i in range(regions_count):
region_crvs = boolean_regions.RegionCurves(i) # type: List[geo.Curve]
result_regions.extend(region_crvs)
return result_regions