-
Notifications
You must be signed in to change notification settings - Fork 28
/
wireframe.py
188 lines (143 loc) · 6.58 KB
/
wireframe.py
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
import numpy as np
def translationMatrix(dx=0, dy=0, dz=0):
""" Return matrix for translation along vector (dx, dy, dz). """
return np.array([[1,0,0,0],
[0,1,0,0],
[0,0,1,0],
[dx,dy,dz,1]])
def translateAlongVectorMatrix(vector, distance):
""" Return matrix for translation along a vector for a given distance. """
unit_vector = np.hstack([unitVector(vector) * distance, 1])
return np.array([[1,0,0,0],[0,1,0,0],[0,0,1,0], unit_vector])
def scaleMatrix(s, cx=0, cy=0, cz=0):
""" Return matrix for scaling equally along all axes centred on the point (cx,cy,cz). """
return np.array([[s,0,0,0],
[0,s,0,0],
[0,0,s,0],
[cx*(1-s), cy*(1-s), cz*(1-s), 1]])
def rotateXMatrix(radians):
""" Return matrix for rotating about the x-axis by 'radians' radians """
c = np.cos(radians)
s = np.sin(radians)
return np.array([[1,0, 0,0],
[0,c,-s,0],
[0,s, c,0],
[0,0, 0,1]])
def rotateYMatrix(radians):
""" Return matrix for rotating about the y-axis by 'radians' radians """
c = np.cos(radians)
s = np.sin(radians)
return np.array([[ c,0,s,0],
[ 0,1,0,0],
[-s,0,c,0],
[ 0,0,0,1]])
def rotateZMatrix(radians):
""" Return matrix for rotating about the z-axis by 'radians' radians """
c = np.cos(radians)
s = np.sin(radians)
return np.array([[c,-s,0,0],
[s, c,0,0],
[0, 0,1,0],
[0, 0,0,1]])
def rotateAboutVector((cx,cy,cz), (x,y,z), radians):
""" Rotate wireframe about given vector by 'radians' radians. """
# Find angle and matrix needed to rotate vector about the z-axis such that its y-component is 0
rotZ = np.arctan2(y, x)
rotZ_matrix = rotateZMatrix(rotZ)
# Find angle and matrix needed to rotate vector about the y-axis such that its x-component is 0
(x, y, z, _) = np.dot(np.array([x,y,z,1]), rotZ_matrix)
rotY = np.arctan2(x, z)
matrix = translationMatrix(dx=-cx, dy=-cy, dz=-cz)
matrix = np.dot(matrix, rotZ_matrix)
matrix = np.dot(matrix, rotateYMatrix(rotY))
matrix = np.dot(matrix, rotateZMatrix(radians))
matrix = np.dot(matrix, rotateYMatrix(-rotY))
matrix = np.dot(matrix, rotateZMatrix(-rotZ))
matrix = np.dot(matrix, translationMatrix(dx=cx, dy=cy, dz=cz))
return matrix
class Wireframe:
""" An array of vectors in R3 and list of edges connecting them. """
def __init__(self, nodes=None):
self.nodes = np.zeros((0,4))
self.edges = []
self.faces = []
if nodes:
self.addNodes(nodes)
def addNodes(self, node_array):
""" Append 1s to a list of 3-tuples and add to self.nodes. """
ones_added = np.hstack((node_array, np.ones((len(node_array),1))))
self.nodes = np.vstack((self.nodes, ones_added))
def addEdges(self, edge_list):
""" Add edges as a list of 2-tuples. """
# Is it better to use a for loop or generate a long list then add it?
# Should raise exception if edge value > len(self.nodes)
self.edges += [edge for edge in edge_list if edge not in self.edges]
def addFaces(self, face_list, face_colour=(255,255,255)):
for node_list in face_list:
num_nodes = len(node_list)
if all((node < len(self.nodes) for node in node_list)):
#self.faces.append([self.nodes[node] for node in node_list])
self.faces.append((node_list, np.array(face_colour, np.uint8)))
self.addEdges([(node_list[n-1], node_list[n]) for n in range(num_nodes)])
def output(self):
if len(self.nodes) > 1:
self.outputNodes()
if self.edges:
self.outputEdges()
if self.faces:
self.outputFaces()
def outputNodes(self):
print "\n --- Nodes --- "
for i, (x, y, z, _) in enumerate(self.nodes):
print " %d: (%d, %d, %d)" % (i, x, y, z)
def outputEdges(self):
print "\n --- Edges --- "
for i, (node1, node2) in enumerate(self.edges):
print " %d: %d -> %d" % (i, node1, node2)
def outputFaces(self):
print "\n --- Faces --- "
for i, nodes in enumerate(self.faces):
print " %d: (%s)" % (i, ", ".join(['%d' % n for n in nodes]))
def transform(self, transformation_matrix):
""" Apply a transformation defined by a transformation matrix. """
self.nodes = np.dot(self.nodes, transformation_matrix)
def findCentre(self):
""" Find the spatial centre by finding the range of the x, y and z coordinates. """
min_values = self.nodes[:,:-1].min(axis=0)
max_values = self.nodes[:,:-1].max(axis=0)
return 0.5*(min_values + max_values)
def sortedFaces(self):
return sorted(self.faces, key=lambda face: min(self.nodes[f][2] for f in face[0]))
def update(self):
""" Override this function to control wireframe behaviour. """
pass
class WireframeGroup:
""" A dictionary of wireframes and methods to manipulate them all together. """
def __init__(self):
self.wireframes = {}
def addWireframe(self, name, wireframe):
self.wireframes[name] = wireframe
def output(self):
for name, wireframe in self.wireframes.items():
print name
wireframe.output()
def outputNodes(self):
for name, wireframe in self.wireframes.items():
print name
wireframe.outputNodes()
def outputEdges(self):
for name, wireframe in self.wireframes.items():
print name
wireframe.outputEdges()
def findCentre(self):
""" Find the central point of all the wireframes. """
# There may be a more efficient way to find the minimums for a group of wireframes
min_values = np.array([wireframe.nodes[:,:-1].min(axis=0) for wireframe in self.wireframes.values()]).min(axis=0)
max_values = np.array([wireframe.nodes[:,:-1].max(axis=0) for wireframe in self.wireframes.values()]).max(axis=0)
return 0.5*(min_values + max_values)
def transform(self, matrix):
for wireframe in self.wireframes.values():
wireframe.transform(matrix)
def update(self):
for wireframe in self.wireframes.values():
wireframe.update()