| 
									
										
											  
											
												Pokémon Red and Blue: Version 4 update (#1963)
## What is this fixing or adding?
Adds a large number of new options, including:
- Door Shuffle
- Sphere-based level scaling
- Key Item and Pokedex requirement options to reach the Elite Four
- Split Card Key option
- Dexsanity option can be set to a percentage of Pokémon that will be checks
- Stonesanity: remove the stones from the Celadon Department Store and shuffle them into the item pool, replacing 4 of the 5 Moon Stone items
- Sleep Trap items option
- Randomize Move Types option
- Town Map Fly Location option, to unlock a flight location when finding/receiving the Town Map
Many enhancements have been made, including:
- Game allows you to continue your save file _from Pallet Town_ as a way to save warp back to the beginning of the game. The one-way drop from Diglett's Cave to north Route 2 that had been added to the randomizer has been removed.
- Client auto-hints some locations when you are able to see the item before you can obtain it (but would only show AP Item if it is for another player), including Bike Shop, Oak's Aides, Celadon Prize Corner, and the unchosen Fossil location.
Various bugs have been fixed, including:
- Route 13 wild Pokémon not correctly logically requiring Cut
- Vanilla tm/hm compatibility options giving compatibility for many TMs/HMs erroneously 
- If an item that exists in multiple quantities in the item pool is chosen for one of the locations that are pre-filled with local items, it will continue placing that same item in the remaining locations as long as more of that item exist
- `start_with` option for `randomize_pokedex` still shuffling a Pokédex into the item pool
- The obedience threshold levels being incorrect with 0-2 badges, with Pokémon up to level 30 obeying with 0-1 badges and up to 10 with 2 badges
- Receiving a DeathLink trigger in the Safari Zone causing issues. Now, you will have your steps remaining set to 0 instead of blacking out when you're in the Safari Zone.
Many location names have been changed, as location names are automatically prepended using the Region name and a large number of areas have been split into new regions as part of the overhaul to add Door Shuffle.
											
										 
											2023-07-23 18:46:54 -04:00
										 |  |  | from . import poke_data | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_surf(state, player): | 
					
						
							|  |  |  |     return (((state.has("HM03 Surf", player) and can_learn_hm(state, "Surf", player)) | 
					
						
							|  |  |  |              or state.has("Flippers", player)) and (state.has("Soul Badge", player) or | 
					
						
							|  |  |  |              state.has(state.multiworld.worlds[player].extra_badges.get("Surf"), player) | 
					
						
							|  |  |  |              or state.multiworld.badges_needed_for_hm_moves[player].value == 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_cut(state, player): | 
					
						
							|  |  |  |     return ((state.has("HM01 Cut", player) and can_learn_hm(state, "Cut", player) or state.has("Master Sword", player)) | 
					
						
							|  |  |  |              and (state.has("Cascade Badge", player) or | 
					
						
							|  |  |  |              state.has(state.multiworld.worlds[player].extra_badges.get("Cut"), player) or | 
					
						
							|  |  |  |              state.multiworld.badges_needed_for_hm_moves[player].value == 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_fly(state, player): | 
					
						
							|  |  |  |     return (((state.has("HM02 Fly", player) and can_learn_hm(state, "Fly", player)) or state.has("Flute", player)) and | 
					
						
							|  |  |  |            (state.has("Thunder Badge", player) or state.has(state.multiworld.worlds[player].extra_badges.get("Fly"), player) | 
					
						
							|  |  |  |             or state.multiworld.badges_needed_for_hm_moves[player].value == 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_strength(state, player): | 
					
						
							|  |  |  |     return ((state.has("HM04 Strength", player) and can_learn_hm(state, "Strength", player)) or | 
					
						
							|  |  |  |             state.has("Titan's Mitt", player)) and (state.has("Rainbow Badge", player) or | 
					
						
							|  |  |  |             state.has(state.multiworld.worlds[player].extra_badges.get("Strength"), player) | 
					
						
							|  |  |  |             or state.multiworld.badges_needed_for_hm_moves[player].value == 0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_flash(state, player): | 
					
						
							|  |  |  |     return (((state.has("HM05 Flash", player) and can_learn_hm(state, "Flash", player)) or state.has("Lamp", player)) | 
					
						
							|  |  |  |              and (state.has("Boulder Badge", player) or state.has(state.multiworld.worlds[player].extra_badges.get("Flash"), | 
					
						
							|  |  |  |              player) or state.multiworld.badges_needed_for_hm_moves[player].value == 0)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_learn_hm(state, move, player): | 
					
						
							|  |  |  |     for pokemon, data in state.multiworld.worlds[player].local_poke_data.items(): | 
					
						
							|  |  |  |         if state.has(pokemon, player) and data["tms"][6] & 1 << (["Cut", "Fly", "Surf", "Strength", | 
					
						
							|  |  |  |                                                                   "Flash"].index(move) + 2): | 
					
						
							|  |  |  |             return True | 
					
						
							|  |  |  |     return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_get_hidden_items(state, player): | 
					
						
							|  |  |  |     return state.has("Item Finder", player) or not state.multiworld.require_item_finder[player].value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def has_key_items(state, count, player): | 
					
						
							|  |  |  |     key_items = (len([item for item in ["Bicycle", "Silph Scope", "Item Finder", "Super Rod", "Good Rod", | 
					
						
							|  |  |  |                                         "Old Rod", "Lift Key", "Card Key", "Town Map", "Coin Case", "S.S. Ticket", | 
					
						
							|  |  |  |                                         "Secret Key", "Poke Flute", "Mansion Key", "Safari Pass", "Plant Key", | 
					
						
							|  |  |  |                                         "Hideout Key", "Card Key 2F", "Card Key 3F", "Card Key 4F", "Card Key 5F", | 
					
						
							|  |  |  |                                         "Card Key 6F", "Card Key 7F", "Card Key 8F", "Card Key 9F", "Card Key 10F", | 
					
						
							|  |  |  |                                         "Card Key 11F", "Exp. All", "Fire Stone", "Thunder Stone", "Water Stone", | 
					
						
							| 
									
										
										
										
											2023-09-10 17:38:56 -04:00
										 |  |  |                                         "Leaf Stone", "Moon Stone"] if state.has(item, player)]) | 
					
						
							| 
									
										
											  
											
												Pokémon Red and Blue: Version 4 update (#1963)
## What is this fixing or adding?
Adds a large number of new options, including:
- Door Shuffle
- Sphere-based level scaling
- Key Item and Pokedex requirement options to reach the Elite Four
- Split Card Key option
- Dexsanity option can be set to a percentage of Pokémon that will be checks
- Stonesanity: remove the stones from the Celadon Department Store and shuffle them into the item pool, replacing 4 of the 5 Moon Stone items
- Sleep Trap items option
- Randomize Move Types option
- Town Map Fly Location option, to unlock a flight location when finding/receiving the Town Map
Many enhancements have been made, including:
- Game allows you to continue your save file _from Pallet Town_ as a way to save warp back to the beginning of the game. The one-way drop from Diglett's Cave to north Route 2 that had been added to the randomizer has been removed.
- Client auto-hints some locations when you are able to see the item before you can obtain it (but would only show AP Item if it is for another player), including Bike Shop, Oak's Aides, Celadon Prize Corner, and the unchosen Fossil location.
Various bugs have been fixed, including:
- Route 13 wild Pokémon not correctly logically requiring Cut
- Vanilla tm/hm compatibility options giving compatibility for many TMs/HMs erroneously 
- If an item that exists in multiple quantities in the item pool is chosen for one of the locations that are pre-filled with local items, it will continue placing that same item in the remaining locations as long as more of that item exist
- `start_with` option for `randomize_pokedex` still shuffling a Pokédex into the item pool
- The obedience threshold levels being incorrect with 0-2 badges, with Pokémon up to level 30 obeying with 0-1 badges and up to 10 with 2 badges
- Receiving a DeathLink trigger in the Safari Zone causing issues. Now, you will have your steps remaining set to 0 instead of blacking out when you're in the Safari Zone.
Many location names have been changed, as location names are automatically prepended using the Region name and a large number of areas have been split into new regions as part of the overhaul to add Door Shuffle.
											
										 
											2023-07-23 18:46:54 -04:00
										 |  |  |                  + min(state.count("Progressive Card Key", player), 10)) | 
					
						
							|  |  |  |     return key_items >= count | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def can_pass_guards(state, player): | 
					
						
							|  |  |  |     if state.multiworld.tea[player]: | 
					
						
							|  |  |  |         return state.has("Tea", player) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return state.has("Vending Machine Drinks", player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def has_badges(state, count, player): | 
					
						
							|  |  |  |     return len([item for item in ["Boulder Badge", "Cascade Badge", "Thunder Badge", "Rainbow Badge", "Marsh Badge", | 
					
						
							|  |  |  |                                   "Soul Badge", "Volcano Badge", "Earth Badge"] if state.has(item, player)]) >= count | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def oaks_aide(state, count, player): | 
					
						
							|  |  |  |     return ((not state.multiworld.require_pokedex[player] or state.has("Pokedex", player)) | 
					
						
							|  |  |  |             and has_pokemon(state, count, player)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def has_pokemon(state, count, player): | 
					
						
							|  |  |  |     obtained_pokemon = set() | 
					
						
							|  |  |  |     for pokemon in poke_data.pokemon_data.keys(): | 
					
						
							|  |  |  |         if state.has(pokemon, player) or state.has(f"Static {pokemon}", player): | 
					
						
							|  |  |  |             obtained_pokemon.add(pokemon) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return len(obtained_pokemon) >= count | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def fossil_checks(state, count, player): | 
					
						
							|  |  |  |     return (state.can_reach('Mt Moon B2F', 'Region', player) and | 
					
						
							|  |  |  |             state.can_reach('Cinnabar Lab Fossil Room', 'Region', player) and | 
					
						
							|  |  |  |             state.can_reach('Cinnabar Island', 'Region', player) and len( | 
					
						
							|  |  |  |         [item for item in ["Dome Fossil", "Helix Fossil", "Old Amber"] if state.has(item, player)]) >= count) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def card_key(state, floor, player): | 
					
						
							|  |  |  |     return state.has(f"Card Key {floor}F", player) or state.has("Card Key", player) or \ | 
					
						
							|  |  |  |            state.has("Progressive Card Key", player, floor - 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def rock_tunnel(state, player): | 
					
						
							|  |  |  |     return can_flash(state, player) or not state.multiworld.dark_rock_tunnel_logic[player] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def route_3(state, player): | 
					
						
							|  |  |  |     if state.multiworld.route_3_condition[player] == "defeat_brock": | 
					
						
							|  |  |  |         return state.has("Defeat Brock", player) | 
					
						
							|  |  |  |     elif state.multiworld.route_3_condition[player] == "defeat_any_gym": | 
					
						
							|  |  |  |         return state.has_any(["Defeat Brock", "Defeat Misty", "Defeat Lt. Surge", "Defeat Erika", "Defeat Koga", | 
					
						
							|  |  |  |                               "Defeat Blaine", "Defeat Sabrina", "Defeat Viridian Gym Giovanni"], player) | 
					
						
							|  |  |  |     elif state.multiworld.route_3_condition[player] == "boulder_badge": | 
					
						
							|  |  |  |         return state.has("Boulder Badge", player) | 
					
						
							|  |  |  |     elif state.multiworld.route_3_condition[player] == "any_badge": | 
					
						
							|  |  |  |         return state.has_any(["Boulder Badge", "Cascade Badge", "Thunder Badge", "Rainbow Badge", "Marsh Badge", | 
					
						
							|  |  |  |                               "Soul Badge", "Volcano Badge", "Earth Badge"], player) | 
					
						
							|  |  |  |     # open | 
					
						
							|  |  |  |     return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def evolve_level(state, level, player): | 
					
						
							|  |  |  |     return len([item for item in ( | 
					
						
							|  |  |  |         "Defeat Brock", "Defeat Misty", "Defeat Lt. Surge", "Defeat Erika", "Defeat Koga", "Defeat Blaine", | 
					
						
							|  |  |  |         "Defeat Sabrina", "Defeat Viridian Gym Giovanni") if state.has(item, player)]) > level / 7 |