-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBarnesHut.py
296 lines (246 loc) · 11.3 KB
/
BarnesHut.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
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
# Barnes hut module
# Gerrit Fritz
import numpy as np
import math
class Node():
"""Stores data for Octree nodes."""
def __init__(self, middle, dimension):
"""Method that sets up a node.
Method that sets up and declares variables for an octree node.
Args:
middle: Position of center of the position.
dimension: Length of sides of a node.
"""
self.particle = None
self.middle = middle
self.dimension = dimension
self.mass = None
self.corners = None
self.center_of_mass = None
self.nodes = []
self.subnodes = [[[None, None], # right front (top, bottom)
[None, None]], # right back (top, bottom)
[[None, None], # left front (top, bottom)
[None, None]]] # left back (top, bottom)
def insert_particle(self, particle):
self.particle = particle
def get_subnode(self, quad):
return (self.subnodes[quad[0]][quad[1]][quad[2]])
def create_subnode(self, quad):
"""Method that creates a subnode.
Method that determines the middle and dimension of the subnode
of a specific quadrant of the node. Initializes a subnode and adds
that subnode to the nodes.
Args:
quad: Quadrant of node.
"""
dimension = self.dimension / 2
x, y, z = 1, 1, 1
if quad[0] == 1:
x = -1
if quad[1] == 1:
y = -1
if quad[2] == 1:
z = -1
middle = [self.middle[0] + ((dimension / 2) * x), # value 1, right
self.middle[1] + ((dimension / 2) * y), # value 1, front
self.middle[2] + ((dimension / 2) * z)] # value 1, top
node = Node(middle, dimension)
self.subnodes[quad[0]][quad[1]][quad[2]] = node
self.nodes.append(node)
def get_quad(self, point):
x, y, z = 1, 1, 1
if point[0] > self.middle[0]: # right
x = 0
if point[1] > self.middle[1]: # front
y = 0
if point[2] > self.middle[2]: # top
z = 0
return [x, y, z]
def get_corners(self):
"""Method that gets corners of a node.
Method that get corners of a node for visualization. Iterates through
the top and bottom for front and back for right and left.
"""
if self.corners is None:
self.corners = []
for x in [1, -1]: # right or left
for y in [1, -1]: # front or back
for z in [1, -1]: # top or bottom
pos = [self.middle[0] + ((self.dimension / 2) * x),
self.middle[1] + ((self.dimension / 2) * y),
self.middle[2] + ((self.dimension / 2) * z)]
self.corners.append(pos)
return self.corners
def in_bounds(self, point):
val = False
if point[0] <= self.middle[0] + (self.dimension / 2) and\
point[0] >= self.middle[0] - (self.dimension / 2) and\
point[1] <= self.middle[1] + (self.dimension / 2) and\
point[1] >= self.middle[1] - (self.dimension / 2) and\
point[2] <= self.middle[2] + (self.dimension / 2) and\
point[2] >= self.middle[2] - (self.dimension / 2):
val = True
return val
def compute_mass_distribution(self):
"""Method that calculates the mass distribution.
Method that calculates the mass distribution of the node based on
the mass posistions of the subnode weighted by weights of
the subnodes.
"""
if self.particle is not None:
self.center_of_mass = np.array([*self.particle.position])
self.mass = self.particle.mass
else:
# Compute the center of mass based on the masses of all child quadrants
# position based on child quadrants weights with their mass
self.mass = 0
self.center_of_mass = np.array([0., 0., 0.])
for node in self.nodes:
if node is not None:
node.compute_mass_distribution()
self.mass += node.mass
self.center_of_mass += node.mass * node.center_of_mass
self.center_of_mass /= self.mass
class Octree():
"""Handles setup and calculations of the Barnes-Hut octree."""
def __init__(self, particles, root_node, theta, node_type):
"""Method that sets up an octree.
Method that sets up the variables for the octree. Calls functions
for creation of the octree.
Args:
particles: List of particles that are inserted.
root_node: Root node of the octree.
theta: Theta that determines the accuracy of the simulations.
"""
self.theta = theta
self.root_node = root_node
self.particles = particles
self.node_type = node_type
for particle in self.particles:
self.insert_to_node(self.root_node, particle)
def insert_to_node(self, node, particle):
"""Recursive method that inserts particles into the octree.
Recursive method that inserts bodies into the octree.
Checks if particle is in the current node to prevent bounds issues.
Determines the appropriate child node and gets that subnode.
If that subnode is empty insert the particle and stop.
If the child node point is a point node (one particle) turn it into
a regional node by inserting both particles into it.
If the child node is a regional node insert the particle.
Args:
node: Quadrant of node.
particle: Simulation body.
"""
# check if point is in cuboid of present node
if not node.in_bounds(particle.position) and not np.array_equal(particle.position, self.root_node.middle):
print("error particle not in bounds")
print(f"middle: {node.middle}, dimension: {node.dimension}, particle position: {particle.position}, type: {type(particle)}")
return
quad = node.get_quad(particle.position)
if node.get_subnode(quad) is None:
node.create_subnode(quad)
subnode = node.get_subnode(quad)
if subnode.particle is None and len(subnode.nodes) == 0: # case empty node
subnode.insert_particle(particle)
elif subnode.particle is not None: # case point node
old_particle = subnode.particle
subnode.insert_particle(None)
self.insert_to_node(subnode, old_particle)
self.insert_to_node(subnode, particle)
elif subnode.particle is None and len(subnode.nodes) >= 1: # case region node
self.insert_to_node(subnode, particle)
def update_forces_collisions(self):
self.collision_dic = {}
for particle in self.particles:
self.collision_dic[particle] = []
particle.force = np.array([0., 0., 0.])
particle.e_pot = 0
self.calc_forces(self.root_node, particle)
particle.e_pot /= 1 # 2
if len(self.collision_dic[particle]) == 0:
del self.collision_dic[particle]
def calc_forces(self, node, particle):
"""Method that calculates the force on an octree particle.
Method that calculates the force on an octree particle by iterating
through the octree.
If the node is a point node that doesnt hold the body itelf, directly
calculate the forces.
If its a regional node and the dimension/distance ratio is smaller
than theta and the center of mass position is not the same as the
particle position, calculate the force between the node and
the particle.
Args:
node: Quadrant of node.
particle: Simulation body.
"""
if node.particle is not None and node.particle != particle:
force, e_pot, distance = self.gravitational_force(particle, node, np.array([]), np.array([]))
particle.force -= force
particle.e_pot -= e_pot
if distance < particle.radius + node.particle.radius and particle.mass > node.particle.mass:
self.collision_dic[particle].append(node.particle)
elif node.particle is None and not np.array_equal(particle.position, node.center_of_mass):
distance = np.array([*particle.position]) - np.array([*node.center_of_mass])
r = math.sqrt(np.dot(distance, distance))
d = node.dimension
if d / r < self.theta:
force, e_pot, distance = self.gravitational_force(particle, node, distance, r)
particle.force -= force
particle.e_pot -= e_pot
else:
for subnode in node.nodes:
self.calc_forces(subnode, particle)
def gravitational_force(self, particle, node, distance_vec, distance_val): # can be ragional or point node
"""Method that calculates the force between two particles.
Method that calculates the force acted on the particle by
another particle or a node. Only calculates the distance and vector
did not have to be calculated for theta.
Args:
particle: Simulation body.
node: Node of the octree.
distance_vec: Vector of distance between the bodies.
distance_val: Magnitude of distance betweent the bodies
Returns:
The force, potential energy and distance between two bodies or
the body and the node.
"""
force = np.array([0., 0., 0.])
if len(distance_vec) == 0 and len(distance_val) == 0:
distance = np.array([*particle.position]) - np.array([*node.center_of_mass])
distance_mag = math.sqrt(np.dot(distance, distance))
else:
distance = distance_vec
distance_mag = distance_val
G = 6.6743 * 10**(-11)
e_pot = G * particle.mass * node.mass / distance_mag
force_mag = G * particle.mass * node.mass / np.dot(distance, distance)
force = (distance / distance_mag) * force_mag
return force, e_pot, distance_mag
def get_all_nodes(self, node, lst):
if node.particle is None and len(node.nodes) >= 1 or node.particle is not None:
if len(node.nodes) >= 1:
if self.node_type == "regional" or self.node_type == "both":
lst.append(node.get_corners())
for subnode in node.nodes:
self.get_all_nodes(subnode, lst)
if node.particle is not None and (self.node_type == "point" or self.node_type == "both"):
lst.append(node.get_corners())
if __name__ == "__main__":
from SolarSystem import Planet
planet1 = Planet
planet1.position = (10, 20, 30)
planet1.mass = 200
planet2 = Planet
planet2.position = (-10, -20, -30)
planet2.mass = 20
data = [planet1, planet2] # planet2]
root = Node([0, 0, 0], 100)
theta = 1
print(root.in_bounds(planet2.position))
quad = root.get_quad(planet2.position)
print(quad)
root.create_subnode(quad)
subnode = root.get_subnode(quad)
print(subnode.middle)
print(subnode.get_corners())