forked from lassefolkersen/highfrontier
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsolarsystem.py
643 lines (626 loc) · 39.9 KB
/
solarsystem.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
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
import SimThread
import os
import math
import sys
from PIL import Image
import pygame
import random
import datetime
import time
import planet
import company
import global_variables
import primitives
import technology
import pickle
from display import Display
sys.setrecursionlimit(10000)
class solarsystem:
display_mode: Display
@property
def display_mode(self) -> Display:
return self._display_mode
@display_mode.setter
def display_mode(self, value: Display | str):
if isinstance(value, str):
value = Display(value)
else:
assert isinstance(value, Display)
self._display_mode = value
def launchThread(self):
self.simThread=SimThread.SimThread(self)
self.simThread.start()
return
def evaluate_each_game_step(self):
"""
Function that is evaluated every single game loop. This is different than the company.evaluate_self functions where
the evaluation frequency is a competition parameter that should be tuneable by selection.
Things to put in here:
Annual base stuff
* migration
* pollution
Perhaps a year report?
"""
#This should only be done once a year - perhaps ok to say once every 360 days ie. every 12 game loops.
try: self.last_evaluation_day_for_evaluate_each_game_step
except:
self.last_evaluation_day_for_evaluate_each_game_step = self.current_date
else:
if self.current_date.year != self.last_evaluation_day_for_evaluate_each_game_step.year:
self.last_evaluation_day_for_evaluate_each_game_step = self.current_date
print_dict = {"text":"running the " + str(self.current_date.year) + " yearly part of evaluate_each_game_step","type":"general gameplay info"}
self.messages.append(print_dict)
self.bitterness_of_world = (None,None)
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
if self.bitterness_of_world[0] is None or self.bitterness_of_world[1] is None:
self.bitterness_of_world = (base.bitternes_of_base, base.bitternes_of_base)
if base.bitternes_of_base > self.bitterness_of_world[1]:
self.bitterness_of_world = (self.bitterness_of_world[0], base.bitternes_of_base)
if base.bitternes_of_base < self.bitterness_of_world[0]:
self.bitterness_of_world = (base.bitternes_of_base, self.bitterness_of_world[1])
#emigration, growth, emission and climate change
for planet in list(self.planets.values()):
planet.check_gas_in_atmosphere()
for base in list(planet.bases.values()):
base.calculate_emigration()
base.calculate_growth_and_deaths()
base.calculate_emissions()
if base.last_mining_check + datetime.timedelta(base.mining_check_interval) < self.current_date:
for resource in self.mineral_resources:
base.get_mining_opportunities(base.home_planet, resource)
people = 0
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
people = people + base.population
number_of_companies = people / global_variables.persons_per_company
if number_of_companies > global_variables.max_number_of_companies:
number_of_companies = global_variables.max_number_of_companies
number_of_companies = number_of_companies - len(self.companies) #do not count the country/companies already in existence
#print "number_of_companies " + str(number_of_companies)
if number_of_companies > 0:
richness_database = {}
for companyName in self.companies:
richness_database[self.companies[companyName].capital] = self.companies[companyName]
richness_list = list(richness_database.keys())
richness_list.sort()
richness_list = richness_list[int(len(richness_list) * 2 / 3):len(richness_list)] #"This is the richest 1/3 of all companies"
list_of_model_companies = []
for key in richness_list:
list_of_model_companies.append(richness_database[key])
#list_of_model_companies_names is just for debugging
list_of_model_companies_names = []
for i in range(0,number_of_companies):
model_company_number = random.randint(0,len(list_of_model_companies)-1)
model_company = list_of_model_companies[model_company_number]
new_company = company.company(self,model_company_database=model_company.company_database,capital = 10000000) #FIXME change capital
self.companies[new_company.name] = new_company
list_of_model_companies_names.append(model_company.name)
print_dict = {"text":"Made " + str(len(list_of_model_companies_names)) + " new companies on the model of: " + str(list_of_model_companies_names),"type":"general company info"}
self.messages.append(print_dict)
def __init__(self,start_date, de_novo_initialization = True):
if de_novo_initialization:
self.display_mode = Display.PLANETARY
self.effectuate_migration = global_variables.effectuate_migration #if False all migration calculations are performed, but are not actually applied to population numbers. Useful for equilibrizing the markets first.
self.effectuate_growth = global_variables.effectuate_growth #if False all growth calculations are performed, but are not actually applied to population numbers. Useful for equilibrizing the markets first.
self.current_player = None #switch that determines what company is playing. If None it would default to simulatormode without possibility of interaction. Else it should be a company object.
self.start_date = start_date
self.current_date = self.start_date
self.step_delay_time = global_variables.step_delay_time # how much delay (in miliseconds, I think) there should be before initiating the next iteration. This can be changed from the settings within the game
self.technology_research_cost = global_variables.technology_research_cost #a variable specifying how much technology costs (in fact it is conversion factor for distance in the technology tree to research points) (100000 is pretty fast)
self.bitterness_of_world = (None,None) # how much bitterness there is in the world. First entry is lowest bitterness level
self.build_base_mode = False #a variable that specifies if a click on the map translates into building a base
self.building_base = None #a link to the building base in build_base_mode - necessary for interplanetary builds
self.messages = []
self.message_printing = {"general gameplay info":True,
"general company info":True,
"tech discovery":False,
"debugging":False,
"base sales":False,
"firm info":True, #this one only goes for the players own firms
"base info":True, #this one only goes for the players own firms
"climate":False,
"mining":False
}
# importing the trade resource text and the mineral_resources
if os.access(os.path.join("data","economy","trade resources.txt"),os.R_OK):
data_file_name = os.path.join("data","economy","trade resources.txt")
trade_resources = primitives.import_datasheet(data_file_name)
mineral_resources = []
for resource_name in trade_resources:
if trade_resources[resource_name]["type"] == "mineral":
mineral_resources.append(resource_name)
self.mineral_resources = mineral_resources
self.trade_resources = trade_resources
else:
raise Exception("Did not find trade_resources.txt file. Warning - this might endanger the integrity of the program")
# self.intialize_globals()
self.window_size = global_variables.window_size
self.universe_creation_date = datetime.date(1969,7,16)
self.current_date = start_date
self.solar_system_zoom = 20480
self.areas_of_interest = {}
self.company_selected = None
self.firm_selected = None
self.go_to_planetary_mode = False
print("initializing planets")
self.planets = self.initialize_planets()
print("done initializing planets")
print("initializing tech tree")
backbone_tree_here = technology.Backbone_Tree()
self.technology_tree = technology.Tree(backbone_tree_here,self)
self.technology_tree.implode_and_expand()
print("done initializing tech tree")
print("initializing companies")
self.companies = self.initialize_companies()
print("done initializing companies")
self.current_planet = self.planets["sun"]
def close_company(self,companyName):
"""
Function that closes down a company by simply deleting it from the list of companies
"""
del self.companies[companyName]
def initialize_planets(self):
data_file_name = os.path.join("data","planets.txt")
read_planet_database = primitives.import_datasheet(data_file_name)
planet_database = {}
for planet_name in read_planet_database:
planet_instance = planet.planet(planet_name,self,read_planet_database[planet_name]) #creating the planet instance
try: global_variables.distance_data
except: global_variables.distance_data = planet_instance.calculate_all_distances()
planet_database[planet_name] = planet_instance
#checking that all projections exist
random_planet_name = random.choice(list(planet_database.keys()))
random_planet = planet_database[random_planet_name]
random_planet.pickle_all_projection_calculations()
return planet_database
def initialize_companies(self):
base_to_country_list = {}
base_to_GNP_list = {}
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
base_to_country_list[base.base_name] = base.original_country
base_to_GNP_list[base.base_name] = base.GDP
country_to_base_list = primitives.invert_dict(base_to_country_list)
country_to_GNP_list = {}
for country in country_to_base_list:
country_GNP = 0
for base in country_to_base_list[country]:
country_GNP = base_to_GNP_list[base] + country_GNP
country_to_GNP_list[country] = country_GNP
### Start up countries from companies.txt in data/economy
data_file_name = os.path.join("data","economy","companies.txt")
read_company_database = primitives.import_datasheet(data_file_name)
random_companyName = random.choice(list(read_company_database.keys()))
for key in read_company_database[random_companyName]:
if len(key) > global_variables.max_letters_in_company_names:
raise Exception(key + " is " + str(len(key)) + " - maximum is " + str(global_variables.max_letters_in_company_names))
#print read_company_database
company_database = {}
for companyName in read_company_database:
company_instance = company.company(self,companyName=companyName,model_company_database=read_company_database[companyName],capital = country_to_GNP_list[companyName]) #creating the company instance
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
if base.base_name in country_to_base_list[companyName]:
company_instance.owned_firms[base.base_name] = base
company_instance.home_cities[base.base_name] = base
company_database[companyName] = company_instance
### starting up all companies (=countries) which are mentioned as owners of a base in the base_data files (and therefore
### put in the self.original_country variable), but which are not found in the above
model_company_data = company_database["united states of america"].company_database
for companyName in country_to_base_list:
if companyName not in company_database:
single_company_data = {}
company_instance = company.company(self,model_company_database=model_company_data,companyName=companyName,capital = country_to_GNP_list[companyName]) #creating the company instance
#company_instance.calculate_company_database(model_company_data, 10)
company_instance.company_database["type"] = "country"
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
if base.base_name in country_to_base_list[companyName]:
company_instance.owned_firms[base.base_name] = base
company_instance.home_cities[base.base_name] = base
# self.firms[base.base_name] = company_instance.home_cities[base.base_name]
company_database[companyName] = company_instance
#### Start up misc private companies
people = []
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
people.append(base.population)
number_of_companies = sum(people) / global_variables.persons_per_company
if number_of_companies > global_variables.max_number_of_companies:
number_of_companies = global_variables.max_number_of_companies
number_of_companies = number_of_companies - len(company_database) #do not count the country/companies already included
GWP = 0
for planet in self.planets:
for base in self.planets[planet].bases:
GWP = self.planets[planet].bases[base].GDP + GWP
mean_capital_per_company = GWP / number_of_companies
all_bases = {}
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
all_bases[base.base_name] = base
for i in range(0,number_of_companies):
#print "starting another company"
capital = random.gauss(mean_capital_per_company,mean_capital_per_company/4)
if capital < 1000:
capital = 1000
home_city = list(all_bases.keys())[random.randint(0,len(all_bases)-1)]
company_instance = company.company(self,capital=capital)
company_instance.company_database["type"] = "private company"
company_instance.home_cities[home_city] = all_bases[home_city]
company_database[company_instance.companyName] = company_instance
#Setting base-ownership for all bases and calculating mining values (this is done now, because all necessary components first are ready now)
for planet in list(self.planets.values()):
for base in list(planet.bases.values()):
base_owner_name = base.original_country
base.owner = company_database[base_owner_name]
for resource in self.mineral_resources + ["food"]:
base.get_mining_opportunities(planet,resource)
#add one of each firm of everything that is known
for company_instance in list(company_database.values()):
for technology in company_instance.known_technologies:
if technology != "common knowledge":
# if random.randint(1,3) == 1:
if True:
start_up_name = technology + " "+ str(random.randint(1000,9999))
size_chosen = random.randint(1, all_bases[home_city].population)
location_name = list(company_instance.home_cities.keys())[0]
location = company_instance.home_cities[location_name]
company_instance.change_firm_size(
location,
size_chosen,
technology,
start_up_name)
start_up_firm = company_instance.owned_firms[start_up_name]
for resource in start_up_firm.input_output_dict["input"]:
start_up_firm.stock_dict[resource] = start_up_firm.input_output_dict["input"][resource]
# print company_instance.name + " started " + start_up_name + " in " + str(location.name)
return company_database
def save_solar_system(self,filename, prepare_scenario = False):
"""
Function that will save the settings. One of the tricky things of doing this is that pictures can not be pickled, so
the function includes the sub-function "get_sub_entry" to find out where pictures are saved. This, however, is not called
by default since it is known where to find the pictures. Pictures are then converted to strings before saving, and
re-converted back after saving. The planet surface pictures that are loaded in memory for faster rendering are removed temporarilu
before the save but inserted afterwards. This means that they are lost on reload, which shouldn't be a big issue.
prepare_for_scenario Kills the current_player
"""
def get_sub_entry(entry,i,before):
"""
This small primitive recursive searcher can be used to check if there are images in the solar system
Suggested use is to call just before the pickle step, to see if the system is clear. However, the call should
be commented out by default since it is really silly extra time to use if it is already known where the pictures are
"""
i = i + 1
if i < 6:
for sub_entry_name in dir(entry):
if sub_entry_name[0:2] != "__":
sub_entry = getattr(entry, sub_entry_name)
if isinstance(sub_entry, instancemethod):
print("found the culprit!")
print(sub_entry_name)
print(sub_entry)
if isinstance(sub_entry, pygame.Surface):
print(str(sub_entry_name) + " is a surface")
try: sub_entry.__class__
except: pass
else:
if issubclass(sub_entry.__class__, Image.Image):
print("The entry_name " + str(sub_entry_name) + " is a " + str(sub_entry.__class__) + " it is found at " + str(before))
before = before + " - " + sub_entry_name
get_sub_entry(sub_entry,i,before)
if prepare_scenario:
print(self)
print(self.current_player)
try: self.current_player.name
except:
print("Preparing for scenario")
else:
raise Exception("Don't prepare for scenario when a player has been started up")
self.current_date = datetime.date(time.localtime()[0],time.localtime()[1],time.localtime()[2]) #this is just for use when creating start out games
self.step_delay_time = global_variables.step_delay_time
#resetting the technologies
for company_instance in list(self.companies.values()):
company_instance.known_technologies = {}
company_instance.pick_research()
company_instance.research = 0
for technology_name in self.technology_tree.vertex_dict:
technology = self.technology_tree.vertex_dict[technology_name]
if technology["known_by"] != "everybody":
technology["known_by"] = {}
#removing pre-drawn surfaces and stringifiyng resource_maps
backup_up_pre_drawn_surfaces = {}
known_planet_images = ["wet_areas","topo_image"]
for planet_instance in list(self.planets.values()):
planet_instance.unload_from_drawing()
for known_planet_image in known_planet_images:
try: getattr(planet_instance, known_planet_image)
except: pass
else:
image = getattr(planet_instance, known_planet_image)
size = image.size
mode = image.mode
image_string = image.tostring()
setattr(planet_instance, known_planet_image, {"string":image_string,"mode":mode,"size":size})
if planet_instance.pre_drawn_surfaces != {}:
backup_up_pre_drawn_surfaces[planet_instance.name] = {}
for pre_drawn_surface_key in planet_instance.pre_drawn_surfaces:
pre_drawn_surface = planet_instance.pre_drawn_surfaces[pre_drawn_surface_key]
backup_up_pre_drawn_surfaces[planet_instance.name][pre_drawn_surface_key] = pre_drawn_surface
planet_instance.pre_drawn_surfaces = {}
#this is where the resource maps gets stringified
if planet_instance.resource_maps != {}:
for resource in planet_instance.resource_maps:
image = planet_instance.resource_maps[resource]
size = image.size
mode = image.mode
image_string = image.tobytes()
planet_instance.resource_maps[resource] = {"string":image_string,"mode":mode,"size":size}
try:
file = open(filename, "wb") # Open the file in binary mode for pickle
if hasattr(self, 'simThread'):
self.simThread.join() # Wait for the simulation thread to finish
del self.simThread # Delete the thread reference
pickle.dump(self, file)
self.launchThread()
else:
pickle.dump(self, file)
print("Game saved successfully.")
except MemoryError:
print("Error: MemoryError occurred while saving the game. Consider freeing up memory or using a system with more memory.")
except pickle.PickleError as e:
print(f"Error: Failed to pickle the game. Details: {e}")
except Exception as e:
print(f"Error: An unexpected error occurred while saving the game. Details: {e}")
finally:
if 'file' in locals() and not file.closed:
file.close()
#here we restore the pre_drawn surfaces
for planet_name in backup_up_pre_drawn_surfaces:
planet_instance = self.planets[planet_name]
for pre_drawn_surface_key in backup_up_pre_drawn_surfaces[planet_name]:
pre_drawn_surface = backup_up_pre_drawn_surfaces[planet_name][pre_drawn_surface_key]
planet_instance.pre_drawn_surfaces[pre_drawn_surface_key] = pre_drawn_surface
#here we de-stringify the resource maps and other known images
for planet_instance in list(self.planets.values()):
for known_planet_image in known_planet_images:
if hasattr(planet_instance, known_planet_image):
image_parts = getattr(planet_instance, known_planet_image)
image = Image.fromstring(image_parts["mode"],image_parts["size"],image_parts["string"])
setattr(planet_instance, known_planet_image, image)
if planet_instance.resource_maps != {}:
for resource in planet_instance.resource_maps:
image_parts = planet_instance.resource_maps[resource]
if image.format:
image = Image.fromstring(image_parts["mode"],image_parts["size"],image_parts["string"])
planet_instance.resource_maps[resource] = image
def load_solar_system(self,filename):
"""
Function that loads the solar system
"""
file = open(filename, "r")
try:
with open(filename, "rb") as file:
new_solar_system = pickle.load(file)
except EOFError as e:
print_dict = {"text": f"Un-loadable file: {filename} - no load performed", "type": "general gameplay info"}
error_message = str(e)
print(error_message)
raise Exception(f"An EOFError of type: {error_message} was found")
except Exception as e:
error_message = str(e)
print(error_message)
raise Exception(f"An error of type: {error_message} was found")
finally:
file.close()
#here we de-stringify the resource maps and other known planet images
known_planet_images = ["wet_areas","topo_image"]
for planet_instance in list(new_solar_system.planets.values()):
for known_planet_image in known_planet_images:
if hasattr(planet_instance, known_planet_image):
image_parts = getattr(planet_instance, known_planet_image)
image = Image.fromstring(image_parts["mode"],image_parts["size"],image_parts["string"])
setattr(planet_instance, known_planet_image, image)
if planet_instance.resource_maps != {}:
for resource in planet_instance.resource_maps:
image_parts = planet_instance.resource_maps[resource]
image = Image.frombytes(image_parts["mode"],image_parts["size"],image_parts["string"])
planet_instance.resource_maps[resource] = image
#inserting all variables in self
for entry_name in dir(self):
if entry_name == '__weakref__':
continue
if not hasattr(new_solar_system, entry_name):
delattr(self,entry_name)
for entry_name in dir(new_solar_system):
if entry_name == '__weakref__':
continue
entry = getattr(new_solar_system, entry_name)
setattr(self,entry_name, entry)
#updating all solar_system_object_links (this is actually rather weird that it has to be done)
for company_instance in list(self.companies.values()):
company_instance.solar_system_object_link = self
for owned_firm_instance in list(company_instance.owned_firms.values()):
owned_firm_instance.solar_system_object_link = self
for planet_instance in list(self.planets.values()):
planet_instance.solars_system_object_link = self
self.technology_tree.solar_system_object_link = self
def get_satellite_to_center_position(self,object,zoom_level,date_variable):
"""
given the orbital parameters and satellite, this returns
the coordinate transposition from the satellite to its center of orbit
"""
orbital_position = self.get_planet_position(object,date_variable)
semi_major_axis = self.planets[object].planet_data["semi_major_axis"] #in km
eccentricity = self.planets[object].planet_data["eccentricity"]
relative_semi_major_axis = float ( semi_major_axis * max(self.window_size) / (17000000000/zoom_level)) #this setting is made so zoom_level = 1 gives a view of all bodies including pluto
relative_semi_minor_axis = ((relative_semi_major_axis ** 2) * (1 - (eccentricity **2) )) ** 0.5
x = relative_semi_major_axis * math.cos(orbital_position - math.pi)
y = relative_semi_minor_axis * math.sin(orbital_position - math.pi)
return (int(-x),int(-y))
def get_planet_position(self,satellite,date_variable):
"""
given the time and planet name, this returns the orbital position as a number between 0 and 2pi
"""
elapsed_time = date_variable - self.universe_creation_date
orbital_period = self.planets[satellite].planet_data["orbital_period_days"]
orbital_position = (int(elapsed_time.days) % orbital_period) / orbital_period
orbital_position = orbital_position * 2 * math.pi
return orbital_position
def get_areas_of_interest(self,zoom_level,date_variable,center_object):
"""
Returns a dictionary of the area of interest of all objects in view, meaning the on-screen coordinate and a some radius
(size_of_target) around it. The keys are the objects orbiting the center_object and all objects with same or
higher hierarchial level
"""
areas_of_interest_here = {}
positions_of_interest = {}
size_of_target = 5
self.window_size
#Determine which orbits to draw
satellite_iterator = self.planets[center_object].planet_data["satellite_of"]
hierarchy_list = [center_object]
while satellite_iterator is not None:
hierarchy_list.append(satellite_iterator)
satellite_iterator = self.planets[satellite_iterator].planet_data["satellite_of"]
#Which orbits to draw - returns the names of all objects that are satellites of the center_object or an object higher in the hierarchy
hierarchy_dependent_transposition = (0,0)
for orbit_center in hierarchy_list:
list = self.calculate_from_a_center(zoom_level,date_variable,surface=None,orbit_center=orbit_center,hierarchy_dependent_transposition=hierarchy_dependent_transposition,draw=False)
positions_of_interest.update(list)
if orbit_center != "sun":
new_hierarchy_dependent_transposition = self.get_satellite_to_center_position(orbit_center,zoom_level,date_variable)
hierarchy_dependent_transposition = (new_hierarchy_dependent_transposition[0] + hierarchy_dependent_transposition[0],new_hierarchy_dependent_transposition[1] + hierarchy_dependent_transposition[1])
#Include the sun
sun_position = (hierarchy_dependent_transposition[0] + global_variables.window_size[0] * 0.5, hierarchy_dependent_transposition[1] + global_variables.window_size[1] * 0.5)
positions_of_interest["sun"] = sun_position
#assign the rectangles
for planet in positions_of_interest:
areas_of_interest_here[(positions_of_interest[planet][0],positions_of_interest[planet][1],size_of_target,size_of_target)] = planet
self.areas_of_interest = areas_of_interest_here
def calculate_from_a_center(self,zoom_level,date_variable,surface,orbit_center,hierarchy_dependent_transposition,draw=True):
"""
This function takes an existing surface, and some orbit-data on a center planet
then adds the orbits and drawings of all the satellites of the center_planet. The drawing can be
moved by using the hierarchy_dependent_transposition variable.
If boolean Draw is false, it will only return a list of positions of the planets
"""
if not draw:
position_list = {}
#print "Info from calculate_from_a_center"
#print "Now drawing from center: " + str(orbit_center)
for object in self.planets:
if self.planets[object].planet_data["satellite_of"] == orbit_center: #draw all the satellites of each of the preceding planets
semi_major_axis = self.planets[object].planet_data["semi_major_axis"] #in km
relative_semi_major_axis = float( semi_major_axis) * max(self.window_size) / (17000000000/zoom_level) #this setting is made so zoom_level = 1 gives a view of all bodies including pluto
eccentricity = self.planets[object].planet_data["eccentricity"]
relative_semi_minor_axis = ((relative_semi_major_axis ** 2) * (1 - (eccentricity **2) )) ** 0.5
type = self.planets[object].planet_type
orbit_color = self.planets[object].planet_data["orbit_color"]
smallplanet_color = self.planets[object].planet_data["smallplanet_color"]
satellite_diameter = self.planets[object].planet_diameter_km * max(self.window_size)*int(zoom_level) / 17000000000
orbital_position = self.get_planet_position(object, date_variable)
orbit_placement = int((orbital_position * 99) / (2 * math.pi))
orbit = []
for i in range(0,100 ):
t = -math.pi + i * ((2 * math.pi) / 100 )
x = self.window_size[0]/2 + relative_semi_major_axis * math.cos(t) + hierarchy_dependent_transposition[0]
y = self.window_size[1]/2 + relative_semi_minor_axis * math.sin(t) + hierarchy_dependent_transposition[1]
orbit.append((x,y))
if draw:
for i in range(0,len(orbit)):
if i == orbit_placement:
x = self.window_size[0]/2 + relative_semi_major_axis * math.cos(orbital_position - math.pi) + hierarchy_dependent_transposition[0]
y = self.window_size[1]/2 + relative_semi_minor_axis * math.sin(orbital_position - math.pi) + hierarchy_dependent_transposition[1]
if 0 < x < self.window_size[0] and 0 < y < self.window_size[1]:
self.draw_small_object(satellite_diameter,(x,y),object,surface)
else:
if -10000 < orbit[i][0] < 10000 and -10000 < orbit[i][1] < 10000:
if i == len(orbit)-1:
pygame.draw.line(surface,(orbit_color),orbit[0],orbit[len(orbit)-1])
elif i in [orbit_placement -1, orbit_placement +1]:
pass
else:
pygame.draw.line(surface,orbit_color,orbit[i],orbit[i+1])
if not draw:
x = self.window_size[0]/2 + relative_semi_major_axis * math.cos(orbital_position - math.pi) + hierarchy_dependent_transposition[0]
y = self.window_size[1]/2 + relative_semi_minor_axis * math.sin(orbital_position - math.pi) + hierarchy_dependent_transposition[1]
if 0 < x < self.window_size[0] and 0 < y < self.window_size[1]:
position_list[object] = (int(x),int(y))
if draw:
return surface
if not draw:
return position_list
def draw_small_object(self,diameter,position,object,surface):
"""
How to draw the individual heavenly bodies
The diameter is a relative diameter corrected for zoom_level as seen in calculate_from_a_center
The position is a tupple of on-screen coordinates
The object is a string with the object name
The surface is a pygame surface on which to draw the small object
These orbital data is used to draw the object as specified on the surface. If zoom is large enough, we go to planetary mode
"""
smallplanet_color = self.planets[object].planet_data["smallplanet_color"]
if self.solar_system_zoom > 2000000:
if object != "sun":
self.display_mode = Display.PLANETARY
self.go_to_planetary_mode = True
else: #this will keep the zoom from going to far in on the sun
self.solar_system_zoom = self.solar_system_zoom / 2
self.draw_small_object(diameter, position, object, surface)
elif diameter < 2.0:
pygame.draw.polygon(surface,smallplanet_color,[position,(position[0],position[1]+1),(position[0]+1,position[1]),(position[0]+1,position[1]+1)])
elif diameter > 20.0 and object != "sun":
if object == self.current_planet.planet_name:
self.display_mode = Display.PLANETARY
self.go_to_planetary_mode = True
else:
if 20 < diameter < 40:
projection_scaling_here = 45
elif 40 <= diameter < 80:
projection_scaling_here = 90
elif 80 <= diameter < 160:
projection_scaling_here = 180
else:
projection_scaling_here = 360
custom_drawn_planet = self.planets[object].draw_image(0,90,projection_scaling_here,fast_rendering=True)
surface.blit(custom_drawn_planet,(position[0]-projection_scaling_here/2,position[1]-projection_scaling_here/2))
else:
pointlist = []
for i in range(1,50):
t = -math.pi + i * ((2 * math.pi) / 50 )
x = position[0] + 0.5* diameter * math.cos(t)
y = position[1] + 0.5* diameter * math.sin(t)
pointlist.append((x,y))
pygame.draw.polygon(surface,smallplanet_color,pointlist)
def draw_solar_system(self,zoom_level,date_variable,center_object):
"""
This is the function that draws the entire solar system
It is a more generalised and experimental version of what is found in the old
draw_solar_system. It still needs to include the sun in the drawing, and also there
possibly is a problem with planets when looking on moons
"""
self.go_to_planetary_mode = False
#Draw the empty space
surface = pygame.Surface(self.window_size)
surface.fill((0,0,0))
#Determine which orbits to draw
satellite_iterator = self.planets[center_object].planet_data["satellite_of"]
hierarchy_list = [center_object]
while satellite_iterator is not None:
hierarchy_list.append(satellite_iterator)
satellite_iterator = self.planets[satellite_iterator].planet_data["satellite_of"]
#Which orbits to draw - returns the names of all objects that are satellites of the center_object or an object higher in the hierarchy
hierarchy_dependent_transposition = (0,0)
for orbit_center in hierarchy_list:
surface = self.calculate_from_a_center(zoom_level,date_variable,surface,orbit_center,hierarchy_dependent_transposition,True)
if orbit_center != "sun":
new_hierarchy_dependent_transposition = self.get_satellite_to_center_position(orbit_center,zoom_level,date_variable)
hierarchy_dependent_transposition = (new_hierarchy_dependent_transposition[0] + hierarchy_dependent_transposition[0],new_hierarchy_dependent_transposition[1] + hierarchy_dependent_transposition[1])
#Drawing the sun
sun_diameter = self.planets["sun"].planet_diameter_km * max(self.window_size)*int(zoom_level) / 17000000000
sun_position = (hierarchy_dependent_transposition[0] + global_variables.window_size[0] * 0.5, hierarchy_dependent_transposition[1] + global_variables.window_size[1] * 0.5)
self.draw_small_object(sun_diameter, sun_position , "sun", surface)
if not self.go_to_planetary_mode:
self.get_areas_of_interest(zoom_level,date_variable,center_object)
return surface