Skip to content

Commit 2249150

Browse files
committed
feat(projection): Add module for projections of all geometry types
1 parent 09488b5 commit 2249150

File tree

3 files changed

+276
-1
lines changed

3 files changed

+276
-1
lines changed

ladybug_geometry/geometry3d/plane.py

+20-1
Original file line numberDiff line numberDiff line change
@@ -333,7 +333,7 @@ def project_point(self, point, projection_direction=None):
333333
"""Project a point onto this Plane given a certain projection direction.
334334
335335
Args:
336-
point: A Point3D to be projected onto the plane
336+
point: A Point3D to be projected onto the plane.
337337
projection_direction: A Line3D or Ray3D object to set the direction
338338
of projection. If None, this Plane's normal will be
339339
used. (Default: None).
@@ -346,6 +346,25 @@ def project_point(self, point, projection_direction=None):
346346
else Ray3D(point, projection_direction)
347347
return intersect_line3d_plane_infinite(int_ray, self)
348348

349+
def project_points(self, points, projection_direction=None):
350+
"""Project an array of points onto this Plane.
351+
352+
Args:
353+
point: A list of Point3Ds to be projected onto the plane.
354+
projection_direction: A Line3D or Ray3D object to set the direction
355+
of projection. If None, this Plane's normal will be
356+
used. (Default: None).
357+
358+
Returns:
359+
A list of Point3Ds for the projected points. This list will include
360+
Nones if the projection_direction is parallel to the plane.
361+
"""
362+
proj_points = []
363+
p_dir = self.n if projection_direction is None else projection_direction
364+
for pt in points:
365+
proj_points.append(intersect_line3d_plane_infinite(Ray3D(pt, p_dir), self))
366+
return proj_points
367+
349368
def intersect_line_ray(self, line_ray):
350369
"""Get the intersection between this plane and the input Line3D or Ray3D.
351370

ladybug_geometry/projection.py

+189
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,189 @@
1+
# coding=utf-8
2+
"""Utility functions for performing plane projections in 3D space.
3+
4+
This module can be used to create axonometric views of geometry among other
5+
purposes.
6+
"""
7+
from __future__ import division
8+
9+
from .geometry2d import Vector2D, Point2D, Ray2D, LineSegment2D, \
10+
Polyline2D, Arc2D, Polygon2D, Mesh2D
11+
from .geometry3d import Vector3D, Point3D, Ray3D, Plane, LineSegment3D, \
12+
Polyline3D, Arc3D, Face3D, Mesh3D, Polyface3D, Sphere, Cone, Cylinder
13+
14+
15+
def project_geometry(plane, geometries):
16+
""""Project multiple geometries into a plane to get them in the world 3D system.
17+
18+
Args:
19+
plane: The Plane into which the geometries will be projected.
20+
geometries: An array of any ladybug_geometry objects which will be projected
21+
into the plane. Note that 2D geometry objects will be converted into
22+
3D (and are assumed to be in the World XY plane) in order to be
23+
projected into the Plane. Also, Arcs will be converted to Polylines
24+
in order to be represented correctly in the plane.
25+
26+
Returns:
27+
The input geometries projected into the Plane. All coordinate values will
28+
be in the World 3D system.
29+
"""
30+
projected_geos = []
31+
for geo in geometries:
32+
if isinstance(geo, (Point3D, Point2D)):
33+
geo = Point3D.from_point2d(geo) if isinstance(geo, Point2D) else geo
34+
projected_geos.append(plane.project_point(geo))
35+
elif isinstance(geo, (LineSegment3D, LineSegment2D)):
36+
geo = LineSegment3D.from_line_segment2d(geo) \
37+
if isinstance(geo, LineSegment2D) else geo
38+
st, end = plane.project_points(geo.endpoints)
39+
projected_geos.append(LineSegment3D.from_end_points(st, end))
40+
elif isinstance(geo, (Polyline3D, Polyline2D)):
41+
geo = Polyline3D.from_polyline2d(geo) if isinstance(geo, Polyline2D) else geo
42+
vertices = plane.project_points(geo.vertices)
43+
projected_geos.append(Polyline3D(vertices, interpolated=geo.interpolated))
44+
elif isinstance(geo, (Arc3D, Arc2D)):
45+
geo = Arc3D.from_arc2d(geo) if isinstance(geo, Arc2D) else geo
46+
p_line = geo.to_polyline(30, interpolated=True)
47+
vertices = plane.project_points(p_line.vertices)
48+
projected_geos.append(Polyline3D(vertices, interpolated=True))
49+
elif isinstance(geo, Polygon2D):
50+
vertices = plane.project_points(geo.vertices)
51+
projected_geos.append(Face3D(vertices))
52+
elif isinstance(geo, Face3D):
53+
boundary = plane.project_points(geo.boundary)
54+
holes = None
55+
if geo.has_holes:
56+
holes = [plane.project_points(h) for h in geo.holes]
57+
projected_geos.append(Face3D(boundary, geo.plane, holes))
58+
elif isinstance(geo, (Mesh3D, Mesh2D)):
59+
geo = Mesh3D.from_mesh2d(geo) if isinstance(geo, Mesh2D) else geo
60+
vertices = plane.project_points(geo.vertices)
61+
projected_geos.append(Mesh3D(vertices, geo.faces, geo.colors))
62+
elif isinstance(geo, Polyface3D):
63+
vertices = plane.project_points(geo.vertices)
64+
proj_p_face = Polyface3D(vertices, geo.face_indices, geo.edge_information)
65+
projected_geos.append(proj_p_face)
66+
elif isinstance(geo, (Ray3D, Ray2D)):
67+
geo = Ray3D.from_ray2d(geo) if isinstance(geo, Ray2D) else geo
68+
pt = plane.project_point(geo.p)
69+
vec = plane.project_point(Point3D(geo.v.x, geo.v.y, geo.v.z))
70+
projected_geos.append(Ray3D(pt, Vector3D(vec.x, vec.y, vec.z)))
71+
elif isinstance(geo, (Vector3D, Vector2D)):
72+
geo = Vector3D.from_vector2d(geo) if isinstance(geo, Vector2D) else geo
73+
vec = plane.project_point(Point3D(geo.x, geo.y, geo.z))
74+
projected_geos.append(Vector3D(vec.x, vec.y, vec.z))
75+
elif isinstance(geo, Plane):
76+
origin = plane.project_point(geo.o)
77+
normal = plane.project_point(Point3D(geo.n.x, geo.n.y, geo.n.z))
78+
normal = Vector3D(normal.x, normal.y, normal.z)
79+
projected_geos.append(Plane(normal, origin))
80+
elif isinstance(geo, Sphere):
81+
center = plane.project_point(geo.center)
82+
projected_geos.append(Sphere(center, geo.radius))
83+
elif isinstance(geo, Cone):
84+
pt = plane.project_point(geo.vertex)
85+
vec = plane.project_point(Point3D(geo.axis.x, geo.axis.y, geo.axis.z))
86+
projected_geos.append(Cone(pt, Vector3D(vec.x, vec.y, vec.z), geo.angle))
87+
elif isinstance(geo, Cylinder):
88+
pt = plane.project_point(geo.center)
89+
vec = plane.project_point(Point3D(geo.axis.x, geo.axis.y, geo.axis.z))
90+
projected_geos.append(Cone(pt, Vector3D(vec.x, vec.y, vec.z), geo.radius))
91+
else:
92+
raise ValueError('Unrecognized geometry type {}: {}'.format(type(geo), geo))
93+
return projected_geos
94+
95+
96+
def project_geometry_2d(plane, geometries):
97+
""""Project multiple geometries into a plane to get them in the plane's 2D system.
98+
99+
Args:
100+
plane: The Plane into which the geometries will be projected.
101+
geometries: An array of any ladybug_geometry objects which will be projected
102+
into the plane. 2D geometry objects will retain their 2D classes as
103+
they are projected into the new system.
104+
105+
Returns:
106+
The input geometries projected into the Plane. All coordinate values will
107+
be in the 2D system of the input plane.
108+
"""
109+
projected_geos = []
110+
for geo in geometries:
111+
if isinstance(geo, (Point3D, Point2D)):
112+
geo = Point3D.from_point2d(geo) if isinstance(geo, Point2D) else geo
113+
projected_geos.append(plane.xyz_to_xy(plane.project_point(geo)))
114+
elif isinstance(geo, (LineSegment3D, LineSegment2D)):
115+
geo = LineSegment3D.from_line_segment2d(geo) \
116+
if isinstance(geo, LineSegment2D) else geo
117+
st, end = plane.project_points(geo.endpoints)
118+
st, end = plane.xyz_to_xy(st), plane.xyz_to_xy(end)
119+
projected_geos.append(LineSegment2D.from_end_points(st, end))
120+
elif isinstance(geo, (Polyline3D, Polyline2D)):
121+
geo = Polyline3D.from_polyline2d(geo) if isinstance(geo, Polyline2D) else geo
122+
vertices = plane.project_points(geo.vertices)
123+
vertices = [plane.xyz_to_xy(pt) for pt in vertices]
124+
projected_geos.append(Polyline2D(vertices, interpolated=geo.interpolated))
125+
elif isinstance(geo, (Arc3D, Arc2D)):
126+
geo = Arc3D.from_arc2d(geo) if isinstance(geo, Arc2D) else geo
127+
p_line = geo.to_polyline(30, interpolated=True)
128+
vertices = plane.project_points(p_line.vertices)
129+
vertices = [plane.xyz_to_xy(pt) for pt in vertices]
130+
projected_geos.append(Polyline2D(vertices, interpolated=True))
131+
elif isinstance(geo, Polygon2D):
132+
vertices = plane.project_points(geo.vertices)
133+
vertices = [plane.xyz_to_xy(pt) for pt in vertices]
134+
projected_geos.append(Polygon2D(vertices))
135+
elif isinstance(geo, Face3D):
136+
boundary = plane.project_points(geo.boundary)
137+
boundary = [Point3D.from_point2d(plane.xyz_to_xy(pt)) for pt in boundary]
138+
holes = None
139+
if geo.has_holes:
140+
holes = []
141+
for h in geo.holes:
142+
h = plane.project_points(h)
143+
h = [Point3D.from_point2d(plane.xyz_to_xy(pt)) for pt in h]
144+
holes.append(h)
145+
projected_geos.append(Face3D(boundary, geo.plane, holes))
146+
elif isinstance(geo, (Mesh3D, Mesh2D)):
147+
geo = Mesh3D.from_mesh2d(geo) if isinstance(geo, Mesh2D) else geo
148+
vertices = plane.project_points(geo.vertices)
149+
vertices = [plane.xyz_to_xy(pt) for pt in vertices]
150+
projected_geos.append(Mesh2D(vertices, geo.faces, geo.colors))
151+
elif isinstance(geo, Polyface3D):
152+
vertices = plane.project_points(geo.vertices)
153+
vertices = [Point3D.from_point2d(plane.xyz_to_xy(pt)) for pt in vertices]
154+
proj_p_face = Polyface3D(vertices, geo.face_indices, geo.edge_information)
155+
projected_geos.append(proj_p_face)
156+
elif isinstance(geo, (Ray3D, Ray2D)):
157+
geo = Ray3D.from_ray2d(geo) if isinstance(geo, Ray2D) else geo
158+
pt = plane.xyz_to_xy(plane.project_point(geo.p))
159+
vec = plane.project_point(Point3D(geo.v.x, geo.v.y, geo.v.z))
160+
vec = plane.xyz_to_xy(vec)
161+
projected_geos.append(Ray2D(pt, Vector2D(vec.x, vec.y)))
162+
elif isinstance(geo, (Vector3D, Vector2D)):
163+
geo = Vector3D.from_vector2d(geo) if isinstance(geo, Vector2D) else geo
164+
vec = plane.project_point(Point3D(geo.x, geo.y, geo.z))
165+
vec = plane.xyz_to_xy(vec)
166+
projected_geos.append(Vector2D(vec.x, vec.y))
167+
elif isinstance(geo, Plane):
168+
origin = plane.xyz_to_xy(plane.project_point(geo.o))
169+
normal = plane.project_point(Point3D(geo.n.x, geo.n.y, geo.n.z))
170+
normal = plane.xyz_to_xy(normal)
171+
normal = Vector3D(normal.x, normal.y)
172+
projected_geos.append(Plane(normal, Point3D(origin.x, origin.y)))
173+
elif isinstance(geo, Sphere):
174+
center = plane.xyz_to_xy(plane.project_point(geo.center))
175+
center = Point3D.from_point2d(center)
176+
projected_geos.append(Sphere(center, geo.radius))
177+
elif isinstance(geo, Cone):
178+
pt = Point3D.from_point2d(plane.xyz_to_xy(plane.project_point(geo.vertex)))
179+
vec = plane.project_point(Point3D(geo.axis.x, geo.axis.y, geo.axis.z))
180+
vec = Point3D.from_point2d(plane.xyz_to_xy(vec))
181+
projected_geos.append(Cone(pt, Vector3D(vec.x, vec.y, vec.z), geo.angle))
182+
elif isinstance(geo, Cylinder):
183+
pt = Point3D.from_point2d(plane.xyz_to_xy(plane.project_point(geo.center)))
184+
vec = plane.project_point(Point3D(geo.axis.x, geo.axis.y, geo.axis.z))
185+
vec = Point3D.from_point2d(plane.xyz_to_xy(vec))
186+
projected_geos.append(Cone(pt, Vector3D(vec.x, vec.y, vec.z), geo.radius))
187+
else:
188+
raise ValueError('Unrecognized geometry type {}: {}'.format(type(geo), geo))
189+
return projected_geos

tests/projection_test.py

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
from ladybug_geometry.geometry2d import LineSegment2D
2+
from ladybug_geometry.geometry3d import Vector3D, Point3D, LineSegment3D, \
3+
Plane, Face3D, Polyface3D
4+
from ladybug_geometry.projection import project_geometry, project_geometry_2d
5+
6+
7+
def test_projection():
8+
"""Test the projection methods with arrays of 3D objects."""
9+
proj_plane = Plane(n=Vector3D(1, 1, 1))
10+
11+
plane1 = Plane(o=Point3D(-5, 0, 0))
12+
plane2 = Plane(o=Point3D(0, -4, 4))
13+
plane3 = Plane(o=Point3D(2, 2, -4))
14+
polyface1 = Polyface3D.from_box(2, 4, 2, plane1)
15+
polyface2 = Polyface3D.from_box(2, 4, 2, plane2)
16+
polyface3 = Polyface3D.from_box(2, 4, 2, plane3)
17+
18+
poly_faces = [polyface1, polyface2, polyface3]
19+
geometries = project_geometry(proj_plane, poly_faces)
20+
assert len(geometries) == 3
21+
for geo in geometries:
22+
assert isinstance(geo, Polyface3D)
23+
assert all(proj_plane.distance_to_point(pt) < 0.001 for pt in geo.vertices)
24+
25+
faces = [f for pf in poly_faces for f in pf.faces]
26+
geometries = project_geometry(proj_plane, faces)
27+
assert len(geometries) == 18
28+
for geo in geometries:
29+
assert isinstance(geo, Face3D)
30+
assert all(proj_plane.distance_to_point(pt) < 0.001 for pt in geo.vertices)
31+
32+
lines = [edge for pf in poly_faces for edge in pf.edges]
33+
geometries = project_geometry(proj_plane, lines)
34+
assert len(geometries) == 36
35+
for geo in geometries:
36+
assert isinstance(geo, LineSegment3D)
37+
assert all(proj_plane.distance_to_point(pt) < 0.001 for pt in geo.vertices)
38+
39+
40+
def test_projection_2d():
41+
"""Test the projection_2d methods with arrays of 3D objects."""
42+
proj_plane = Plane(n=Vector3D(1, 1, 1))
43+
44+
plane1 = Plane(o=Point3D(-5, 0, 0))
45+
plane2 = Plane(o=Point3D(0, -4, 4))
46+
plane3 = Plane(o=Point3D(2, 2, -4))
47+
polyface1 = Polyface3D.from_box(2, 4, 2, plane1)
48+
polyface2 = Polyface3D.from_box(2, 4, 2, plane2)
49+
polyface3 = Polyface3D.from_box(2, 4, 2, plane3)
50+
51+
poly_faces = [polyface1, polyface2, polyface3]
52+
geometries = project_geometry_2d(proj_plane, poly_faces)
53+
assert len(geometries) == 3
54+
for geo in geometries:
55+
assert isinstance(geo, Polyface3D)
56+
57+
faces = [f for pf in poly_faces for f in pf.faces]
58+
geometries = project_geometry_2d(proj_plane, faces)
59+
assert len(geometries) == 18
60+
for geo in geometries:
61+
assert isinstance(geo, Face3D)
62+
63+
lines = [edge for pf in poly_faces for edge in pf.edges]
64+
geometries = project_geometry_2d(proj_plane, lines)
65+
assert len(geometries) == 36
66+
for geo in geometries:
67+
assert isinstance(geo, LineSegment2D)

0 commit comments

Comments
 (0)