-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathplanet.py
102 lines (87 loc) · 3.7 KB
/
planet.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
import pygame
import math
from decimal import Decimal
class Planet:
# Constants
AU = 149.6e6 * 1000 # Astronomical Unit in meters
G = 6.67428e-11 # Gravitational constant
TIMESTEP = 3600 * 24 # Time step for simulation (1 day in seconds)
# Class variables
_planet_count = 0 # Counter for the number of planets
_initial_scale = 100 / AU # Initial scale factor for drawing
def __init__(self, x, y, radius, color, mass, isSun):
"""
Initialize a new planet with position, size, color, mass, and whether it is the Sun.
"""
self.x = x
self.y = y
self.radius = radius
self.color = color
self.mass = mass
self.sun = isSun
self.distance_to_sun = 0
self.orbit = []
self.x_vel = 0
self.y_vel = 0
# Update the scale based on the number of planets
Planet._planet_count += 1
Planet.SCALE = Planet._initial_scale * (1 / (Planet._planet_count ** 0.9)) # Adjust scaling factor
def draw(self, win):
"""
Draw the planet and its orbit on the provided window surface.
"""
# Compute the screen coordinates for the planet
x = self.x * Planet.SCALE + win.get_width() / 2
y = self.y * Planet.SCALE + win.get_height() / 2
# Draw the orbit if available
if len(self.orbit) > 2:
updated_points = []
for point in self.orbit:
px, py = point
px = px * Planet.SCALE + win.get_width() / 2
py = py * Planet.SCALE + win.get_height() / 2
updated_points.append((px, py))
pygame.draw.lines(win, self.color, False, updated_points, 2)
# Draw the planet itself
# Scale the radius based on the number of planets
pygame.draw.circle(win, self.color, (int(x), int(y)), int(self.radius / Planet._planet_count ** 2.98))
# Display the distance from the Sun if it's not the Sun
if not self.sun:
FONT = pygame.font.SysFont("comicsans", 16) # Create font
distance_text = FONT.render(f"{'%.2E' % Decimal(self.distance_to_sun / 1000)} km", 1, (255, 255, 255))
win.blit(distance_text, (x - distance_text.get_width() / 2, y - distance_text.get_height() / 2))
def attraction(self, other):
"""
Calculate the gravitational force exerted on this planet by another planet.
"""
other_x, other_y = other.x, other.y
distance_x = other_x - self.x
distance_y = other_y - self.y
distance = math.sqrt(distance_x ** 2 + distance_y ** 2)
# Update the distance to the Sun if the other planet is the Sun
if other.sun:
self.distance_to_sun = distance
# Calculate gravitational force
force = self.G * self.mass * other.mass / distance ** 2
theta = math.atan2(distance_y, distance_x)
force_x = math.cos(theta) * force
force_y = math.sin(theta) * force
return force_x, force_y
def update_position(self, planets):
"""
Update the planet's velocity and position based on gravitational forces from other planets.
"""
total_fx = total_fy = 0
for planet in planets:
if self == planet:
continue
fx, fy = self.attraction(planet)
total_fx += fx
total_fy += fy
# Update velocity and position
self.x_vel += total_fx / self.mass * self.TIMESTEP
self.y_vel += total_fy / self.mass * self.TIMESTEP
self.x += self.x_vel * self.TIMESTEP
self.y += self.y_vel * self.TIMESTEP
# Append the new position to the orbit list
self.orbit.append((self.x, self.y))