Skip to content

Commit b797f22

Browse files
committed
add demo script
1 parent db6f08c commit b797f22

File tree

1 file changed

+180
-0
lines changed

1 file changed

+180
-0
lines changed

src/demo.py

+180
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,180 @@
1+
"""This demo script renders a test scene in non-euclidean geometry."""
2+
3+
from enum import IntEnum
4+
5+
from nerte.coordinates import Coordinates
6+
from nerte.vector import Vector
7+
from nerte.face import Face
8+
from nerte.object import Object
9+
from nerte.camera import Camera
10+
from nerte.scene import Scene
11+
from nerte.color import RandomColorDispenser
12+
from nerte.geometry import Geometry, DummyNonEuclideanGeometry
13+
from nerte.renderer import ImageRenderer
14+
15+
16+
class Axis(IntEnum):
17+
"""represents an axis"""
18+
19+
X = 0
20+
Y = 1
21+
Z = 2
22+
23+
24+
class Side(IntEnum):
25+
"""represents a half (triangle) of a square"""
26+
27+
THIS = -1
28+
THAT = +1
29+
30+
31+
class Distance(IntEnum):
32+
"""represents the negative or positive range of an axis"""
33+
34+
NEAR = -1
35+
FAR = +1
36+
37+
38+
# pseudo-random color generator
39+
COLOR = RandomColorDispenser()
40+
41+
42+
def make_camera(canvas_dimension: int) -> Camera:
43+
"""creates a camera with preset values"""
44+
45+
location = Coordinates(0.0, 0.0, -2.0)
46+
direction = Vector(0.0, 0.0, 1.0)
47+
width_vec = Vector(1.0, 0.0, 0.0)
48+
height_vec = Vector(0.0, 1.0, 0.0)
49+
camera = Camera(
50+
location=location,
51+
direction=direction,
52+
canvas_dimensions=(canvas_dimension, canvas_dimension),
53+
detector_manifold=(width_vec, height_vec),
54+
)
55+
return camera
56+
57+
58+
def make_triangle_object(fix: Axis, distance: Distance, side: Side) -> Object:
59+
"""creates a section of a cube (triangle) where each section gets assigned a random color"""
60+
61+
# intermediate matrix for coordinate coefficients
62+
coords = [[None for _ in range(3)] for _ in range(3)]
63+
# create the coefficients based on the parameters
64+
for coord in coords:
65+
coord[fix.value] = 1.0 * distance.value
66+
axis_u, axis_v = (axis for axis in (0, 1, 2) if axis != fix.value)
67+
coords[0][axis_u] = -1.0
68+
coords[0][axis_v] = -1.0
69+
coords[1][axis_u] = -1.0 * side.value
70+
coords[1][axis_v] = +1.0 * side.value
71+
coords[2][axis_u] = +1.0
72+
coords[2][axis_v] = +1.0
73+
# represent the coefficients as proper coordinates
74+
point0 = Coordinates(*coords[0])
75+
point1 = Coordinates(*coords[1])
76+
point2 = Coordinates(*coords[2])
77+
# create the triangle as an object
78+
tri = Face(point0, point1, point2)
79+
obj = Object(color=next(COLOR)) # pseudo-random color
80+
obj.add_face(tri)
81+
return obj
82+
83+
84+
def make_scene(canvas_dimension: int) -> Scene:
85+
"""creates a scene with a camera pointing inside a cube with no front face"""
86+
87+
camera = make_camera(canvas_dimension)
88+
scene = Scene(camera=camera)
89+
90+
# add all faces of the hollow cube as separate object to enable
91+
# individual colors for each triange
92+
# object 1
93+
obj = make_triangle_object(Axis.Y, Distance.NEAR, Side.THAT)
94+
scene.add_object(obj)
95+
# object 2
96+
obj = make_triangle_object(Axis.Y, Distance.NEAR, Side.THIS)
97+
scene.add_object(obj)
98+
# object 3
99+
obj = make_triangle_object(Axis.Z, Distance.FAR, Side.THAT)
100+
scene.add_object(obj)
101+
# object 4
102+
obj = make_triangle_object(Axis.Z, Distance.FAR, Side.THIS)
103+
scene.add_object(obj)
104+
# object 5
105+
obj = make_triangle_object(Axis.X, Distance.NEAR, Side.THAT)
106+
scene.add_object(obj)
107+
# object 6
108+
obj = make_triangle_object(Axis.X, Distance.NEAR, Side.THIS)
109+
scene.add_object(obj)
110+
# object 7
111+
obj = make_triangle_object(Axis.X, Distance.FAR, Side.THAT)
112+
scene.add_object(obj)
113+
# object 8
114+
obj = make_triangle_object(Axis.X, Distance.FAR, Side.THIS)
115+
scene.add_object(obj)
116+
# object 9
117+
obj = make_triangle_object(Axis.Y, Distance.FAR, Side.THAT)
118+
scene.add_object(obj)
119+
# object 10
120+
obj = make_triangle_object(Axis.Y, Distance.FAR, Side.THIS)
121+
scene.add_object(obj)
122+
# NOTE: There are no triangles for Axis.Z and Distance.NEAR since they
123+
# would cover up the inside of the cube.
124+
125+
return scene
126+
127+
128+
def render(
129+
scene: Scene,
130+
geometry: Geometry,
131+
output_path: str,
132+
file_prefix: str,
133+
show: bool,
134+
):
135+
"""renders a preset scene with non-euclidean geometry in orthographic and
136+
perspective projection"""
137+
138+
print("rendering orthographic projection ...")
139+
image_renderer = ImageRenderer(
140+
mode=ImageRenderer.Mode.ORTHOGRAPHIC,
141+
)
142+
image_renderer.render(scene=scene, geometry=geometry)
143+
image_renderer.save(path=f"{output_path}/{file_prefix}_ortho.png")
144+
if show:
145+
image_renderer.show()
146+
147+
print("rendering perspective projection ...")
148+
image_renderer = ImageRenderer(
149+
mode=ImageRenderer.Mode.PERSPECTIVE,
150+
)
151+
image_renderer.render(scene=scene, geometry=geometry)
152+
image_renderer.save(path=f"{output_path}/{file_prefix}_persp.png")
153+
if show:
154+
image_renderer.show()
155+
156+
157+
def main():
158+
"""creates and renders the demo scene"""
159+
160+
# NOTE: Increase the canvas dimension to improve the image quality.
161+
# This will also increase rendering time!
162+
scene = make_scene(canvas_dimension=100)
163+
# NOTE: max_ray_length must be long enough to reach all surfaces.
164+
# NOTE: max_steps controlls the accuracy of the approximation
165+
# NOTE: Increase the bend_factor to increase the 'swirl' effect.
166+
# bend_factor=0.0 results in euclidean geometry.
167+
geo = DummyNonEuclideanGeometry(max_steps=16, max_ray_length=10.0, bend_factor=0.4)
168+
169+
# NOTE: Set show to False if images cannot be displayed.
170+
render(
171+
scene=scene,
172+
geometry=geo,
173+
output_path="../images",
174+
file_prefix="res",
175+
show=True,
176+
)
177+
178+
179+
if __name__ == "__main__":
180+
main()

0 commit comments

Comments
 (0)