Particle_Simulator/Python_Stuff/universe.py

154 lines
7.3 KiB
Python
Raw Normal View History

2024-11-23 21:59:28 -05:00
import numpy as np
DELTA_TIME = 0.05
GRAV_CONST = 6.674 # m^3*s^2/kg
class universe():
def __init__(self):
self.matA = None
self.matB = None
self.matC = None
self.line_array = []
self.mass_array = []
self.radius_array = []
self.force_direction = []
self.positions = []
self.velocities = []
self.accelerations = []
self.particle_list = []
self.sprites = []
print("New universe has been created")
def __repr__(self):
return "Your universe contains the following particles: " +str(self.particle_list)
def _recalculate_matA(self):
#
# | 0 m2 m3 m4 |
# | |
# | m1 0 m3 m4 |
# A = | | * G
# | m1 m2 0 m4 |
# | |
# | m1 m2 m3 0 |
# | |
#
temp_array = np.matrix(self.mass_array)
self.matA = np.matmul(np.ones(temp_array.T.shape), temp_array)
self.matA = GRAV_CONST * self.matA
self.matA = self.matA - np.multiply(self.matA, np.eye(*self.matA.shape))
print(self.matA)
def _recalculate_matB(self):
# | 1 [x1-x2]^2 [x1-x3]^2 [x1-x4]^2 |
# | |
# | [x2-x1]^2 1 [x2-x3]^2 [x2-x4]^2 |
# B = | |
# | [x3-x1]^2 [x3-x2]^2 1 [x3-x4]^2 |
# | |
# | [x4-x1]^2 [x4-x2]^2 [x4-x3]^2 1 |
# | |
temp_array = np.tile(np.array(self.positions), (len(self.positions), 1, 1))
self.force_direction = temp_array - np.transpose(temp_array, axes=(1, 0, 2))
self.matB = np.empty(self.force_direction.shape[0:2])
for counti, i in enumerate(self.force_direction):
for countj, j in enumerate(i):
self.matB[counti, countj] = np.dot(j, j)
if self.matB[counti, countj] == 0.0:
self.matB[counti, countj] = 1.0
self.force_direction[counti, countj] = j / np.sqrt(self.matB[counti, countj])
def _recalculate_matC(self):
# | 0 [r1+r2]^2 [r1+r3]^2 [r1+r4]^2 |
# | |
# | [r2+r1]^2 0 [r2+r3]^2 [r2+r4]^2 |
# C = | |
# | [r3+r1]^2 [r3+r2]^2 0 [r3+r4]^2 |
# | |
# | [r4+r1]^2 [r4+r2]^2 [r4+r3]^2 0 |
# | |
temp_array = np.matrix(self.radius_array)
self.matC = np.matmul(np.ones(temp_array.T.shape), temp_array) + temp_array.T
self.matC = np.multiply(self.matC, self.matC)
# print("Matrix C:", self.matC)
def add_particle(self, particle):
self.mass_array.append(particle.mass)
self.positions.append(particle.position)
self.velocities.append(particle.velocity)
self.accelerations.append(particle.acceleration)
self.particle_list.append(particle)
self.radius_array.append(particle.radius)
self.sprites.append(particle.sprite)
# print("Added new particle")
self._recalculate_matA()
self._recalculate_matB()
self._recalculate_matC()
def add_line(self, line):
self.line_array.append(line)
# def process_collisions(self):
# def update_gravity(self):
def process(self):
# Apply gravity
temp = self.matA/self.matB
for counti in range(temp.shape[0]):
accel = np.array([0.0, 10.0])
for countj in range(temp.shape[1]):
accel += temp[counti, countj] * self.force_direction[counti, countj]
self.accelerations[counti] = accel
self.velocities[counti] = self.accelerations[counti] * DELTA_TIME + self.velocities[counti]
self.positions[counti] = self.velocities[counti] * DELTA_TIME + self.positions[counti]
self.sprites[counti].update_pos(self.positions[counti]-self.radius_array[counti])
self._recalculate_matB()
# Check for collisions...
for counti in range(temp.shape[0]):
# ...with other particles
for countj in range(counti+1, temp.shape[0]):
if (self.matB[counti, countj] <= self.matC[counti, countj]):
direction_vector = self.positions[countj] - self.positions[counti]
dist = np.linalg.norm(direction_vector)
direction_vector = direction_vector / dist
temp_vel_1 = np.dot(self.velocities[counti], direction_vector)*(direction_vector)
leftover_vel_1 = self.velocities[counti] - temp_vel_1
temp_vel_2 = np.dot(self.velocities[countj], direction_vector)*(direction_vector)
leftover_vel_2 = self.velocities[countj] - temp_vel_2
m1_2 = self.mass_array[counti]*2
m2_2 = self.mass_array[countj]*2
m1_plus_m2 = self.mass_array[counti]+self.mass_array[countj]
m1_minus_m2 = self.mass_array[counti] - self.mass_array[countj]
self.velocities[countj] = (0.9999*m1_2*temp_vel_1-m1_minus_m2*temp_vel_2)/m1_plus_m2 + leftover_vel_2
self.velocities[counti] = (0.9999*m1_minus_m2*temp_vel_1+m2_2*temp_vel_2)/m1_plus_m2 + leftover_vel_1
# self.positions[countj] = self.velocities[countj] * DELTA_TIME + self.positions[countj]
# self.positions[counti] = self.velocities[counti] * DELTA_TIME + self.positions[counti]
self.positions[countj] = self.positions[countj] + direction_vector*((np.sqrt(self.matC[counti,countj])-np.sqrt(self.matB[counti,countj]))*self.mass_array[counti]/m1_plus_m2)
self.positions[counti] = self.positions[counti] - direction_vector*((np.sqrt(self.matC[counti,countj])-np.sqrt(self.matB[counti,countj]))*self.mass_array[countj]/m1_plus_m2)
# with lines
for line in self.line_array:
vec1 = line.position - np.array(self.positions[counti])
dist = np.dot(vec1, line.vector)*line.vector
norm = np.linalg.norm(dist)
direction = dist/norm
if(norm < self.radius_array[counti]):
# print("HIT!")
speed = np.dot(self.velocities[counti], line.vector)*line.vector
self.velocities[counti] = self.velocities[counti] - 2*np.linalg.norm(speed)*direction
# self.positions[counti] = self.velocities[counti] * DELTA_TIME + self.positions[counti]
self.positions[counti] = self.positions[counti]-direction*(self.radius_array[counti]-norm)