| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  | from typing import Dict, List, Set, TYPE_CHECKING | 
					
						
							| 
									
										
										
										
											2021-12-18 13:01:24 +01:00
										 |  |  | from collections import deque | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-18 01:33:40 +02:00
										 |  |  | from .Options import TechTreeLayout | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  | if TYPE_CHECKING: | 
					
						
							|  |  |  |     from . import Factorio, FactorioScienceLocation | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | funnel_layers = {TechTreeLayout.option_small_funnels: 3, | 
					
						
							|  |  |  |                  TechTreeLayout.option_medium_funnels: 4, | 
					
						
							|  |  |  |                  TechTreeLayout.option_large_funnels: 5} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | funnel_slice_sizes = {TechTreeLayout.option_small_funnels: 6, | 
					
						
							|  |  |  |                       TechTreeLayout.option_medium_funnels: 10, | 
					
						
							|  |  |  |                       TechTreeLayout.option_large_funnels: 15} | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-04 22:21:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  | def _sorter(location: "FactorioScienceLocation"): | 
					
						
							|  |  |  |     return location.complexity, location.rel_cost | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_shapes(factorio_world: "Factorio") -> Dict["FactorioScienceLocation", Set["FactorioScienceLocation"]]: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     world = factorio_world.multiworld | 
					
						
							| 
									
										
										
										
											2021-06-11 18:02:48 +02:00
										 |  |  |     player = factorio_world.player | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |     prerequisites: Dict["FactorioScienceLocation", Set["FactorioScienceLocation"]] = {} | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  |     layout = world.tech_tree_layout[player].value | 
					
						
							| 
									
										
										
										
											2023-04-09 20:58:24 +02:00
										 |  |  |     locations: List["FactorioScienceLocation"] = sorted(factorio_world.science_locations, key=lambda loc: loc.name) | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |     world.random.shuffle(locations) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-18 13:05:43 +01:00
										 |  |  |     if layout == TechTreeLayout.option_single: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     elif layout == TechTreeLayout.option_small_diamonds: | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  |         slice_size = 4 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  |             diamond_0, diamond_1, diamond_2, diamond_3 = slice | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             #   0    | | 
					
						
							|  |  |  |             # 1   2  | | 
					
						
							|  |  |  |             #   3    V | 
					
						
							|  |  |  |             prerequisites[diamond_3] = {diamond_1, diamond_2} | 
					
						
							|  |  |  |             prerequisites[diamond_2] = prerequisites[diamond_1] = {diamond_0} | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  |     elif layout == TechTreeLayout.option_medium_diamonds: | 
					
						
							|  |  |  |         slice_size = 9 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #     0     | | 
					
						
							|  |  |  |             #   1   2   | | 
					
						
							|  |  |  |             # 3   4   5 | | 
					
						
							|  |  |  |             #   6   7   | | 
					
						
							|  |  |  |             #     8     V | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1]} | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[1], slice[2]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[2]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[6]] = {slice[3], slice[4]} | 
					
						
							|  |  |  |             prerequisites[slice[7]] = {slice[4], slice[5]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[8]] = {slice[6], slice[7]} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |     elif layout == TechTreeLayout.option_large_diamonds: | 
					
						
							|  |  |  |         slice_size = 16 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #       0       | | 
					
						
							|  |  |  |             #     1   2     | | 
					
						
							|  |  |  |             #   3   4   5   | | 
					
						
							|  |  |  |             # 6   7   8   9 | | 
					
						
							|  |  |  |             #   10  11  12  | | 
					
						
							|  |  |  |             #     13  14    | | 
					
						
							|  |  |  |             #       15      | | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1]} | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[1], slice[2]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[2]} | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |             prerequisites[slice[6]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[7]] = {slice[3], slice[4]} | 
					
						
							|  |  |  |             prerequisites[slice[8]] = {slice[4], slice[5]} | 
					
						
							|  |  |  |             prerequisites[slice[9]] = {slice[5]} | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |             prerequisites[slice[10]] = {slice[6], slice[7]} | 
					
						
							|  |  |  |             prerequisites[slice[11]] = {slice[7], slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[12]] = {slice[8], slice[9]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[13]] = {slice[10], slice[11]} | 
					
						
							|  |  |  |             prerequisites[slice[14]] = {slice[11], slice[12]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[15]] = {slice[13], slice[14]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif layout == TechTreeLayout.option_small_pyramids: | 
					
						
							|  |  |  |         slice_size = 6 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #        0       | | 
					
						
							|  |  |  |             #      1   2     | | 
					
						
							|  |  |  |             #    3   4   5   | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1]} | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[1], slice[2]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[2]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif layout == TechTreeLayout.option_medium_pyramids: | 
					
						
							|  |  |  |         slice_size = 10 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #        0       | | 
					
						
							|  |  |  |             #      1   2     | | 
					
						
							|  |  |  |             #    3   4   5   | | 
					
						
							|  |  |  |             #  6   7   8   9 | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1]} | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[1], slice[2]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[2]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[6]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[7]] = {slice[3], slice[4]} | 
					
						
							|  |  |  |             prerequisites[slice[8]] = {slice[4], slice[5]} | 
					
						
							|  |  |  |             prerequisites[slice[9]] = {slice[5]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif layout == TechTreeLayout.option_large_pyramids: | 
					
						
							|  |  |  |         slice_size = 15 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             #         0          | | 
					
						
							|  |  |  |             #       1   2        | | 
					
						
							|  |  |  |             #     3   4   5      | | 
					
						
							|  |  |  |             #   6   7   8   9    | | 
					
						
							|  |  |  |             # 10  11  12  13  14 | | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1]} | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[1], slice[2]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[2]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[6]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[7]] = {slice[3], slice[4]} | 
					
						
							|  |  |  |             prerequisites[slice[8]] = {slice[4], slice[5]} | 
					
						
							|  |  |  |             prerequisites[slice[9]] = {slice[5]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[10]] = {slice[6]} | 
					
						
							|  |  |  |             prerequisites[slice[11]] = {slice[6], slice[7]} | 
					
						
							|  |  |  |             prerequisites[slice[12]] = {slice[7], slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[13]] = {slice[8], slice[9]} | 
					
						
							|  |  |  |             prerequisites[slice[14]] = {slice[9]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     elif layout in funnel_layers: | 
					
						
							|  |  |  |         slice_size = funnel_slice_sizes[layout] | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         world.random.shuffle(locations) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             current_locations = locations[:slice_size] | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |             layer_size = funnel_layers[layout] | 
					
						
							|  |  |  |             previous_slice = [] | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |             current_locations.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |             for layer in range(funnel_layers[layout]): | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |                 slice = current_locations[:layer_size] | 
					
						
							|  |  |  |                 current_locations = current_locations[layer_size:] | 
					
						
							| 
									
										
										
										
											2021-06-06 20:26:40 +02:00
										 |  |  |                 if previous_slice: | 
					
						
							|  |  |  |                     for i, tech_name in enumerate(slice): | 
					
						
							|  |  |  |                         prerequisites.setdefault(tech_name, set()).update(previous_slice[i:i+2]) | 
					
						
							|  |  |  |                 previous_slice = slice | 
					
						
							|  |  |  |                 layer_size -= 1 | 
					
						
							| 
									
										
										
										
											2021-12-18 13:01:24 +01:00
										 |  |  |     elif layout == TechTreeLayout.option_trees: | 
					
						
							|  |  |  |         #              0              | | 
					
						
							|  |  |  |         #            1   2            | | 
					
						
							|  |  |  |         #              3              | | 
					
						
							|  |  |  |         #        4   5   6   7        | | 
					
						
							|  |  |  |         #              8              | | 
					
						
							|  |  |  |         #  9   10   11   12   13  14  | | 
					
						
							|  |  |  |         #              15             | | 
					
						
							|  |  |  |         #              16             | | 
					
						
							|  |  |  |         slice_size = 17 | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         while len(locations) > slice_size: | 
					
						
							|  |  |  |             slice = locations[:slice_size] | 
					
						
							|  |  |  |             locations = locations[slice_size:] | 
					
						
							|  |  |  |             slice.sort(key=_sorter) | 
					
						
							| 
									
										
										
										
											2021-12-18 13:01:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[1]] = {slice[0]} | 
					
						
							|  |  |  |             prerequisites[slice[2]] = {slice[0]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[3]] = {slice[1], slice[2]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[4]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[5]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[6]] = {slice[3]} | 
					
						
							|  |  |  |             prerequisites[slice[7]] = {slice[3]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[8]] = {slice[4], slice[5], slice[6], slice[7]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[9]] = {slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[10]] = {slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[11]] = {slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[12]] = {slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[13]] = {slice[8]} | 
					
						
							|  |  |  |             prerequisites[slice[14]] = {slice[8]} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prerequisites[slice[15]] = {slice[9], slice[10], slice[11], slice[12], slice[13], slice[14]} | 
					
						
							|  |  |  |             prerequisites[slice[16]] = {slice[15]} | 
					
						
							|  |  |  |     elif layout == TechTreeLayout.option_choices: | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |         locations.sort(key=_sorter) | 
					
						
							|  |  |  |         current_choices = deque([locations[0]]) | 
					
						
							|  |  |  |         locations = locations[1:] | 
					
						
							|  |  |  |         while len(locations) > 1: | 
					
						
							| 
									
										
										
										
											2021-12-18 13:01:24 +01:00
										 |  |  |             source = current_choices.pop() | 
					
						
							| 
									
										
										
										
											2022-10-28 21:00:06 +02:00
										 |  |  |             choices = locations[:2] | 
					
						
							|  |  |  |             locations = locations[2:] | 
					
						
							| 
									
										
										
										
											2021-12-18 13:01:24 +01:00
										 |  |  |             for choice in choices: | 
					
						
							|  |  |  |                 prerequisites[choice] = {source} | 
					
						
							|  |  |  |             current_choices.extendleft(choices) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         raise NotImplementedError(f"Layout {layout} is not implemented.") | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-24 01:58:26 +02:00
										 |  |  |     factorio_world.tech_tree_layout_prerequisites = prerequisites | 
					
						
							| 
									
										
										
										
											2021-04-11 18:19:47 +02:00
										 |  |  |     return prerequisites |