| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | from __future__ import annotations | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | from typing import Optional, Union, List, Tuple, Callable, Dict, TYPE_CHECKING | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | from Fill import FillError | 
					
						
							| 
									
										
										
										
											2022-10-12 13:28:32 -05:00
										 |  |  | from .Options import LTTPBosses as Bosses | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | from .StateHelpers import can_shoot_arrows, can_extend_magic, can_get_good_bee, has_sword, has_beam_sword, \ | 
					
						
							| 
									
										
										
										
											2024-02-19 19:07:49 -05:00
										 |  |  |     has_melee_weapon, has_fire_source, can_use_bombs | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | if TYPE_CHECKING: | 
					
						
							|  |  |  |     from . import ALTTPWorld | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Boss: | 
					
						
							|  |  |  |     def __init__(self, name: str, enemizer_name: str, defeat_rule: Callable, player: int): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.enemizer_name = enemizer_name | 
					
						
							|  |  |  |         self.defeat_rule = defeat_rule | 
					
						
							|  |  |  |         self.player = player | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def can_defeat(self, state) -> bool: | 
					
						
							|  |  |  |         return self.defeat_rule(state, self.player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return f"Boss({self.name})" | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-26 10:48:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | def BossFactory(boss: str, player: int) -> Optional[Boss]: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     if boss in boss_table: | 
					
						
							|  |  |  |         enemizer_name, defeat_rule = boss_table[boss] | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |         return Boss(boss, enemizer_name, defeat_rule, player) | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |     raise Exception('Unknown Boss: %s', boss) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def ArmosKnightsDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     # Magic amounts are probably a bit overkill | 
					
						
							|  |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             has_melee_weapon(state, player) or | 
					
						
							|  |  |  |             can_shoot_arrows(state, player) or | 
					
						
							|  |  |  |             (state.has('Cane of Somaria', player) and can_extend_magic(state, player, 10)) or | 
					
						
							|  |  |  |             (state.has('Cane of Byrna', player) and can_extend_magic(state, player, 16)) or | 
					
						
							|  |  |  |             (state.has('Ice Rod', player) and can_extend_magic(state, player, 32)) or | 
					
						
							|  |  |  |             (state.has('Fire Rod', player) and can_extend_magic(state, player, 32)) or | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |             state.has('Blue Boomerang', player) or | 
					
						
							|  |  |  |             state.has('Red Boomerang', player)) | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def LanmolasDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             has_melee_weapon(state, player) or | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |             state.has('Fire Rod', player) or | 
					
						
							|  |  |  |             state.has('Ice Rod', player) or | 
					
						
							|  |  |  |             state.has('Cane of Somaria', player) or | 
					
						
							|  |  |  |             state.has('Cane of Byrna', player) or | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             can_shoot_arrows(state, player)) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def MoldormDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     return has_melee_weapon(state, player) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def HelmasaurKingDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2020-03-15 14:14:06 +11:00
										 |  |  |     # TODO: technically possible with the hammer | 
					
						
							| 
									
										
										
										
											2024-02-19 19:07:49 -05:00
										 |  |  |     return (can_use_bombs(state, player, 5) or state.has("Hammer", player)) and (has_sword(state, player) | 
					
						
							|  |  |  |                                                                                  or can_shoot_arrows(state, player)) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def ArrghusDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |     if not state.has('Hookshot', player): | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         return False | 
					
						
							|  |  |  |     # TODO: ideally we would have a check for bow and silvers, which combined with the | 
					
						
							|  |  |  |     # hookshot is enough. This is not coded yet because the silvers that only work in pyramid feature | 
					
						
							|  |  |  |     # makes this complicated | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     if has_melee_weapon(state, player): | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     return ((state.has('Fire Rod', player) and (can_shoot_arrows(state, player) or can_extend_magic(state, player, | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |                                                                                                          12))) or  # assuming mostly gitting two puff with one shot | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             (state.has('Ice Rod', player) and (can_shoot_arrows(state, player) or can_extend_magic(state, player, 16)))) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def MothulaDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             has_melee_weapon(state, player) or | 
					
						
							|  |  |  |             (state.has('Fire Rod', player) and can_extend_magic(state, player, 10)) or | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |             # TODO: Not sure how much (if any) extend magic is needed for these two, since they only apply | 
					
						
							|  |  |  |             # to non-vanilla locations, so are harder to test, so sticking with what VT has for now: | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |             (state.has('Cane of Somaria', player) and can_extend_magic(state, player, 16)) or | 
					
						
							|  |  |  |             (state.has('Cane of Byrna', player) and can_extend_magic(state, player, 16)) or | 
					
						
							|  |  |  |             can_get_good_bee(state, player) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def BlindDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     return has_melee_weapon(state, player) or state.has('Cane of Somaria', player) or state.has('Cane of Byrna', player) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def KholdstareDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     return ( | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |             ( | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |                     state.has('Fire Rod', player) or | 
					
						
							|  |  |  |                     ( | 
					
						
							|  |  |  |                             state.has('Bombos', player) and | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                             (has_sword(state, player) or state.multiworld.swordless[player]) | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |                     ) | 
					
						
							|  |  |  |             ) and | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |             ( | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                     has_melee_weapon(state, player) or | 
					
						
							|  |  |  |                     (state.has('Fire Rod', player) and can_extend_magic(state, player, 20)) or | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |                     ( | 
					
						
							|  |  |  |                             state.has('Fire Rod', player) and | 
					
						
							|  |  |  |                             state.has('Bombos', player) and | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |                             state.multiworld.swordless[player] and | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                             can_extend_magic(state, player, 16) | 
					
						
							| 
									
										
										
										
											2020-04-20 19:17:10 +02:00
										 |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |             ) | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def VitreousDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2025-01-13 02:32:59 -05:00
										 |  |  |     return ((can_shoot_arrows(state, player) and can_use_bombs(state, player, 10)) | 
					
						
							|  |  |  |             or can_shoot_arrows(state, player, 35) or state.has("Silver Bow", player) | 
					
						
							|  |  |  |             or has_melee_weapon(state, player)) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def TrinexxDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2019-04-18 11:23:24 +02:00
										 |  |  |     if not (state.has('Fire Rod', player) and state.has('Ice Rod', player)): | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         return False | 
					
						
							| 
									
										
										
										
											2020-03-15 14:14:06 +11:00
										 |  |  |     return state.has('Hammer', player) or state.has('Tempered Sword', player) or state.has('Golden Sword', player) or \ | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |            (state.has('Master Sword', player) and can_extend_magic(state, player, 16)) or \ | 
					
						
							|  |  |  |            (has_sword(state, player) and can_extend_magic(state, player, 32)) | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def AgahnimDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     return has_sword(state, player) or state.has('Hammer', player) or state.has('Bug Catching Net', player) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | def GanonDefeatRule(state, player: int) -> bool: | 
					
						
							| 
									
										
										
										
											2022-10-31 21:41:21 -05:00
										 |  |  |     if state.multiworld.swordless[player]: | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |         return state.has('Hammer', player) and \ | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                has_fire_source(state, player) and \ | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |                state.has('Silver Bow', player) and \ | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                can_shoot_arrows(state, player) | 
					
						
							| 
									
										
										
										
											2020-12-04 22:44:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |     can_hurt = has_beam_sword(state, player) | 
					
						
							|  |  |  |     common = can_hurt and has_fire_source(state, player) | 
					
						
							| 
									
										
										
										
											2021-06-14 22:10:26 -05:00
										 |  |  |     # silverless ganon may be needed in anything higher than no glitches | 
					
						
							| 
									
										
										
										
											2024-02-19 19:07:49 -05:00
										 |  |  |     if state.multiworld.glitches_required[player] != 'no_glitches': | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  |         # need to light torch a sufficient amount of times | 
					
						
							|  |  |  |         return common and (state.has('Tempered Sword', player) or state.has('Golden Sword', player) or ( | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |                 state.has('Silver Bow', player) and can_shoot_arrows(state, player)) or | 
					
						
							|  |  |  |                            state.has('Lamp', player) or can_extend_magic(state, player, 12)) | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2023-03-03 23:23:52 -08:00
										 |  |  |         return common and state.has('Silver Bow', player) and can_shoot_arrows(state, player) | 
					
						
							| 
									
										
										
										
											2020-10-07 19:51:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | boss_table: Dict[str, Tuple[str, Optional[Callable]]] = { | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     'Armos Knights': ('Armos', ArmosKnightsDefeatRule), | 
					
						
							|  |  |  |     'Lanmolas': ('Lanmola', LanmolasDefeatRule), | 
					
						
							|  |  |  |     'Moldorm': ('Moldorm', MoldormDefeatRule), | 
					
						
							|  |  |  |     'Helmasaur King': ('Helmasaur', HelmasaurKingDefeatRule), | 
					
						
							|  |  |  |     'Arrghus': ('Arrghus', ArrghusDefeatRule), | 
					
						
							|  |  |  |     'Mothula': ('Mothula', MothulaDefeatRule), | 
					
						
							|  |  |  |     'Blind': ('Blind', BlindDefeatRule), | 
					
						
							|  |  |  |     'Kholdstare': ('Kholdstare', KholdstareDefeatRule), | 
					
						
							|  |  |  |     'Vitreous': ('Vitreous', VitreousDefeatRule), | 
					
						
							|  |  |  |     'Trinexx': ('Trinexx', TrinexxDefeatRule), | 
					
						
							|  |  |  |     'Agahnim': ('Agahnim', AgahnimDefeatRule), | 
					
						
							|  |  |  |     'Agahnim2': ('Agahnim2', AgahnimDefeatRule) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | boss_location_table: List[Tuple[str, str]] = [ | 
					
						
							| 
									
										
										
										
											2021-03-26 04:05:36 +01:00
										 |  |  |         ('Ganons Tower', 'top'), | 
					
						
							|  |  |  |         ('Tower of Hera', None), | 
					
						
							|  |  |  |         ('Skull Woods', None), | 
					
						
							|  |  |  |         ('Ganons Tower', 'middle'), | 
					
						
							|  |  |  |         ('Eastern Palace', None), | 
					
						
							|  |  |  |         ('Desert Palace', None), | 
					
						
							|  |  |  |         ('Palace of Darkness', None), | 
					
						
							|  |  |  |         ('Swamp Palace', None), | 
					
						
							|  |  |  |         ('Thieves Town', None), | 
					
						
							|  |  |  |         ('Ice Palace', None), | 
					
						
							|  |  |  |         ('Misery Mire', None), | 
					
						
							|  |  |  |         ('Turtle Rock', None), | 
					
						
							|  |  |  |         ('Ganons Tower', 'bottom'), | 
					
						
							| 
									
										
										
										
											2021-03-07 04:36:46 -08:00
										 |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 00:07:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | def place_plando_bosses(world: "ALTTPWorld", bosses: List[str]) -> Tuple[List[str], List[Tuple[str, str]]]: | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     # Most to least restrictive order | 
					
						
							|  |  |  |     boss_locations = boss_location_table.copy() | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |     world.multiworld.random.shuffle(boss_locations) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     boss_locations.sort(key=lambda location: -int(restrictive_boss_locations[location])) | 
					
						
							|  |  |  |     already_placed_bosses: List[str] = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for boss in bosses: | 
					
						
							|  |  |  |         if "-" in boss:  # handle plando locations | 
					
						
							|  |  |  |             loc, boss = boss.split("-") | 
					
						
							|  |  |  |             boss = boss.title() | 
					
						
							|  |  |  |             level: str = None | 
					
						
							|  |  |  |             if loc.split(" ")[-1] in {"top", "middle", "bottom"}: | 
					
						
							|  |  |  |                 # split off level | 
					
						
							|  |  |  |                 loc = loc.split(" ") | 
					
						
							|  |  |  |                 level = loc[-1] | 
					
						
							|  |  |  |                 loc = " ".join(loc[:-1]) | 
					
						
							|  |  |  |             loc = loc.title().replace("Of", "of") | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             place_boss(world, boss, loc, level) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |             already_placed_bosses.append(boss) | 
					
						
							|  |  |  |             boss_locations.remove((loc, level)) | 
					
						
							|  |  |  |         else:  # boss chosen with no specified locations | 
					
						
							|  |  |  |             boss = boss.title() | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             boss_locations, already_placed_bosses = place_where_possible(world, boss, boss_locations) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return already_placed_bosses, boss_locations | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  | def can_place_boss(boss: str, dungeon_name: str, level: Optional[str] = None) -> bool: | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |     # blacklist approach | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |     if boss in {"Agahnim", "Agahnim2", "Ganon"}: | 
					
						
							|  |  |  |         return False | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |     if dungeon_name == 'Ganons Tower': | 
					
						
							|  |  |  |         if level == 'top': | 
					
						
							|  |  |  |             if boss in {"Armos Knights", "Arrghus", "Blind", "Trinexx", "Lanmolas"}: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         elif level == 'middle': | 
					
						
							|  |  |  |             if boss == "Blind": | 
					
						
							|  |  |  |                 return False | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |     elif dungeon_name == 'Tower of Hera': | 
					
						
							|  |  |  |         if boss in {"Armos Knights", "Arrghus", "Blind", "Trinexx", "Lanmolas"}: | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |     elif dungeon_name == 'Skull Woods': | 
					
						
							|  |  |  |         if boss == "Trinexx": | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | restrictive_boss_locations: Dict[Tuple[str, str], bool] = {} | 
					
						
							| 
									
										
										
										
											2021-03-26 04:05:36 +01:00
										 |  |  | for location in boss_location_table: | 
					
						
							|  |  |  |     restrictive_boss_locations[location] = not all(can_place_boss(boss, *location) | 
					
						
							|  |  |  |                                                for boss in boss_table if not boss.startswith("Agahnim")) | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | def place_boss(world: "ALTTPWorld", boss: str, location: str, level: Optional[str]) -> None: | 
					
						
							|  |  |  |     player = world.player | 
					
						
							|  |  |  |     if location == 'Ganons Tower' and world.multiworld.mode[player] == 'inverted': | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |         location = 'Inverted Ganons Tower' | 
					
						
							| 
									
										
										
										
											2020-12-31 13:25:14 +01:00
										 |  |  |     logging.debug('Placing boss %s at %s', boss, location + (' (' + level + ')' if level else '')) | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |     world.dungeons[location].bosses[level] = BossFactory(boss, player) | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | def format_boss_location(location_name: str, level: str) -> str: | 
					
						
							|  |  |  |     return location_name + (' (' + level + ')' if level else '') | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | def place_bosses(world: "ALTTPWorld") -> None: | 
					
						
							|  |  |  |     multiworld = world.multiworld | 
					
						
							|  |  |  |     player = world.player | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     # will either be an int or a lower case string with ';' between options | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |     boss_shuffle: Union[str, int] = multiworld.boss_shuffle[player].value | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     already_placed_bosses: List[str] = [] | 
					
						
							|  |  |  |     remaining_locations: List[Tuple[str, str]] = [] | 
					
						
							|  |  |  |     # handle plando | 
					
						
							|  |  |  |     if isinstance(boss_shuffle, str): | 
					
						
							|  |  |  |         # figure out our remaining mode, convert it to an int and remove it from plando_args | 
					
						
							|  |  |  |         options = boss_shuffle.split(";") | 
					
						
							|  |  |  |         boss_shuffle = Bosses.options[options.pop()] | 
					
						
							|  |  |  |         # place our plando bosses | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |         already_placed_bosses, remaining_locations = place_plando_bosses(world, options) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     if boss_shuffle == Bosses.option_none:  # vanilla boss locations | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |         return | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     # Most to least restrictive order | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     if not remaining_locations and not already_placed_bosses: | 
					
						
							|  |  |  |         remaining_locations = boss_location_table.copy() | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |     multiworld.random.shuffle(remaining_locations) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     remaining_locations.sort(key=lambda location: -int(restrictive_boss_locations[location])) | 
					
						
							| 
									
										
										
										
											2019-08-18 15:22:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |     all_bosses = sorted(boss_table.keys())  # sorted to be deterministic on older pythons | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |     placeable_bosses = [boss for boss in all_bosses if boss not in ['Agahnim', 'Agahnim2', 'Ganon']] | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     if boss_shuffle == Bosses.option_basic or boss_shuffle == Bosses.option_full: | 
					
						
							|  |  |  |         if boss_shuffle == Bosses.option_basic:  # vanilla bosses shuffled | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |             bosses = placeable_bosses + ['Armos Knights', 'Lanmolas', 'Moldorm'] | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |         else:  # all bosses present, the three duplicates chosen at random | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             bosses = placeable_bosses + multiworld.random.sample(placeable_bosses, 3) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |         # there is probably a better way to do this | 
					
						
							|  |  |  |         while already_placed_bosses: | 
					
						
							|  |  |  |             # remove already manually placed bosses, to prevent for example triple Lanmolas | 
					
						
							|  |  |  |             boss = already_placed_bosses.pop() | 
					
						
							|  |  |  |             if boss in bosses: | 
					
						
							|  |  |  |                 bosses.remove(boss) | 
					
						
							|  |  |  |             # there may be more bosses than locations at this point, depending on manual placement | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 00:07:55 +02:00
										 |  |  |         logging.debug('Bosses chosen %s', bosses) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |         multiworld.random.shuffle(bosses) | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |         for loc, level in remaining_locations: | 
					
						
							| 
									
										
										
										
											2021-03-26 04:05:36 +01:00
										 |  |  |             for _ in range(len(bosses)): | 
					
						
							|  |  |  |                 boss = bosses.pop() | 
					
						
							|  |  |  |                 if can_place_boss(boss, loc, level): | 
					
						
							|  |  |  |                     break | 
					
						
							|  |  |  |                 # put the boss back in queue | 
					
						
							|  |  |  |                 bosses.insert(0, boss)  # this would be faster with deque, | 
					
						
							|  |  |  |                 # but the deque size is small enough that it should not matter | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise FillError(f'Could not place boss for location {format_boss_location(loc, level)}') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             place_boss(world, boss, loc, level) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     elif boss_shuffle == Bosses.option_chaos:  # all bosses chosen at random | 
					
						
							|  |  |  |         for loc, level in remaining_locations: | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |                 boss = multiworld.random.choice( | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |                     [b for b in placeable_bosses if can_place_boss(b, loc, level)]) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  |             except IndexError: | 
					
						
							| 
									
										
										
										
											2021-03-26 04:05:36 +01:00
										 |  |  |                 raise FillError(f'Could not place boss for location {format_boss_location(loc, level)}') | 
					
						
							| 
									
										
										
										
											2020-07-31 21:56:31 +02:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |                 place_boss(world, boss, loc, level) | 
					
						
							| 
									
										
										
										
											2018-09-26 13:12:20 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     elif boss_shuffle == Bosses.option_singularity: | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |         primary_boss = multiworld.random.choice(placeable_bosses) | 
					
						
							|  |  |  |         remaining_boss_locations, _ = place_where_possible(world, primary_boss, remaining_locations) | 
					
						
							| 
									
										
										
										
											2020-08-19 21:10:02 +02:00
										 |  |  |         if remaining_boss_locations: | 
					
						
							|  |  |  |             # pick a boss to go into the remaining locations | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             remaining_boss = multiworld.random.choice([boss for boss in placeable_bosses if all( | 
					
						
							| 
									
										
										
										
											2020-11-22 11:39:20 +01:00
										 |  |  |                 can_place_boss(boss, loc, level) for loc, level in remaining_boss_locations)]) | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             remaining_boss_locations, _ = place_where_possible(world, remaining_boss, remaining_boss_locations) | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |             if remaining_boss_locations: | 
					
						
							|  |  |  |                 raise Exception("Unfilled boss locations!") | 
					
						
							| 
									
										
										
										
											2020-08-25 23:53:15 +02:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |         raise FillError(f"Could not find boss shuffle mode {boss_shuffle}") | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  | def place_where_possible(world: "ALTTPWorld", boss: str, boss_locations) -> Tuple[List[Tuple[str, str]], List[str]]: | 
					
						
							| 
									
										
										
										
											2022-09-16 19:55:33 -05:00
										 |  |  |     remainder: List[Tuple[str, str]] = [] | 
					
						
							|  |  |  |     placed_bosses: List[str] = [] | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |     for loc, level in boss_locations: | 
					
						
							|  |  |  |         # place that boss where it can go | 
					
						
							|  |  |  |         if can_place_boss(boss, loc, level): | 
					
						
							| 
									
										
										
										
											2023-05-20 19:57:48 +02:00
										 |  |  |             place_boss(world, boss, loc, level) | 
					
						
							| 
									
										
										
										
											2020-12-31 13:23:32 +01:00
										 |  |  |             placed_bosses.append(boss) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             remainder.append((loc, level)) | 
					
						
							|  |  |  |     return remainder, placed_bosses |