252 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			252 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			Python
		
	
	
	
| from typing import Dict, List, Set, TYPE_CHECKING
 | |
| from collections import deque
 | |
| 
 | |
| from .Options import TechTreeLayout
 | |
| 
 | |
| if TYPE_CHECKING:
 | |
|     from . import Factorio, FactorioScienceLocation
 | |
| 
 | |
| 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}
 | |
| 
 | |
| 
 | |
| def _sorter(location: "FactorioScienceLocation"):
 | |
|     return location.complexity, location.rel_cost
 | |
| 
 | |
| 
 | |
| def get_shapes(factorio_world: "Factorio") -> Dict["FactorioScienceLocation", Set["FactorioScienceLocation"]]:
 | |
|     world = factorio_world.multiworld
 | |
|     player = factorio_world.player
 | |
|     prerequisites: Dict["FactorioScienceLocation", Set["FactorioScienceLocation"]] = {}
 | |
|     layout = world.tech_tree_layout[player].value
 | |
|     locations: List["FactorioScienceLocation"] = sorted(factorio_world.science_locations, key=lambda loc: loc.name)
 | |
|     world.random.shuffle(locations)
 | |
| 
 | |
|     if layout == TechTreeLayout.option_single:
 | |
|         pass
 | |
|     elif layout == TechTreeLayout.option_small_diamonds:
 | |
|         slice_size = 4
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
|             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}
 | |
| 
 | |
|     elif layout == TechTreeLayout.option_medium_diamonds:
 | |
|         slice_size = 9
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             #     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]}
 | |
| 
 | |
|     elif layout == TechTreeLayout.option_large_diamonds:
 | |
|         slice_size = 16
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             #       0       |
 | |
|             #     1   2     |
 | |
|             #   3   4   5   |
 | |
|             # 6   7   8   9 |
 | |
|             #   10  11  12  |
 | |
|             #     13  14    |
 | |
|             #       15      |
 | |
| 
 | |
|             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], 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
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             #        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
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             #        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
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             #         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]
 | |
|         world.random.shuffle(locations)
 | |
| 
 | |
|         while len(locations) > slice_size:
 | |
|             locations = locations[slice_size:]
 | |
|             current_locations = locations[:slice_size]
 | |
|             layer_size = funnel_layers[layout]
 | |
|             previous_slice = []
 | |
|             current_locations.sort(key=_sorter)
 | |
|             for layer in range(funnel_layers[layout]):
 | |
|                 slice = current_locations[:layer_size]
 | |
|                 current_locations = current_locations[layer_size:]
 | |
|                 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
 | |
|     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
 | |
|         while len(locations) > slice_size:
 | |
|             slice = locations[:slice_size]
 | |
|             locations = locations[slice_size:]
 | |
|             slice.sort(key=_sorter)
 | |
| 
 | |
|             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:
 | |
|         locations.sort(key=_sorter)
 | |
|         current_choices = deque([locations[0]])
 | |
|         locations = locations[1:]
 | |
|         while len(locations) > 1:
 | |
|             source = current_choices.pop()
 | |
|             choices = locations[:2]
 | |
|             locations = locations[2:]
 | |
|             for choice in choices:
 | |
|                 prerequisites[choice] = {source}
 | |
|             current_choices.extendleft(choices)
 | |
|     else:
 | |
|         raise NotImplementedError(f"Layout {layout} is not implemented.")
 | |
| 
 | |
|     factorio_world.tech_tree_layout_prerequisites = prerequisites
 | |
|     return prerequisites
 |