| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | import itertools | 
					
						
							|  |  |  | from collections import Counter | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | from typing import NamedTuple, TYPE_CHECKING | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  | from BaseClasses import Item, ItemClassification | 
					
						
							|  |  |  | from .options import BossesAsChecks, VictoryCondition, ExtraOrbs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if TYPE_CHECKING: | 
					
						
							|  |  |  |     from . import NoitaWorld | 
					
						
							|  |  |  | else: | 
					
						
							|  |  |  |     NoitaWorld = object | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ItemData(NamedTuple): | 
					
						
							| 
									
										
										
										
											2023-07-18 22:51:01 -04:00
										 |  |  |     code: int | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     group: str | 
					
						
							|  |  |  |     classification: ItemClassification = ItemClassification.progression | 
					
						
							|  |  |  |     required_num: int = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class NoitaItem(Item): | 
					
						
							|  |  |  |     game: str = "Noita" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def create_item(player: int, name: str) -> Item: | 
					
						
							|  |  |  |     item_data = item_table[name] | 
					
						
							|  |  |  |     return NoitaItem(name, item_data.classification, item_data.code, player) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | def create_fixed_item_pool() -> list[str]: | 
					
						
							|  |  |  |     required_items: dict[str, int] = {name: data.required_num for name, data in item_table.items()} | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     return list(Counter(required_items).elements()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | def create_orb_items(victory_condition: VictoryCondition, extra_orbs: ExtraOrbs) -> list[str]: | 
					
						
							| 
									
										
										
										
											2023-07-18 22:51:01 -04:00
										 |  |  |     orb_count = extra_orbs.value | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     if victory_condition == VictoryCondition.option_pure_ending: | 
					
						
							| 
									
										
										
										
											2023-07-18 22:51:01 -04:00
										 |  |  |         orb_count = orb_count + 11 | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     elif victory_condition == VictoryCondition.option_peaceful_ending: | 
					
						
							| 
									
										
										
										
											2023-07-18 22:51:01 -04:00
										 |  |  |         orb_count = orb_count + 33 | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     return ["Orb" for _ in range(orb_count)] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | def create_spatial_awareness_item(bosses_as_checks: BossesAsChecks) -> list[str]: | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     return ["Spatial Awareness Perk"] if bosses_as_checks.value >= BossesAsChecks.option_all_bosses else [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | def create_kantele(victory_condition: VictoryCondition) -> list[str]: | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     return ["Kantele"] if victory_condition.value >= VictoryCondition.option_pure_ending else [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | def create_random_items(world: NoitaWorld, weights: dict[str, int], count: int) -> list[str]: | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |     filler_pool = weights.copy() | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |     if not world.options.bad_effects: | 
					
						
							| 
									
										
										
										
											2025-02-23 11:27:05 -05:00
										 |  |  |         filler_pool["Trap"] = 0 | 
					
						
							|  |  |  |         filler_pool["Greed Die"] = 0 | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |     return world.random.choices(population=list(filler_pool.keys()), | 
					
						
							|  |  |  |                                 weights=list(filler_pool.values()), | 
					
						
							|  |  |  |                                 k=count) | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  | def create_all_items(world: NoitaWorld) -> None: | 
					
						
							|  |  |  |     player = world.player | 
					
						
							|  |  |  |     locations_to_fill = len(world.multiworld.get_unfilled_locations(player)) | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     itempool = ( | 
					
						
							|  |  |  |         create_fixed_item_pool() | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |         + create_orb_items(world.options.victory_condition, world.options.extra_orbs) | 
					
						
							|  |  |  |         + create_spatial_awareness_item(world.options.bosses_as_checks) | 
					
						
							|  |  |  |         + create_kantele(world.options.victory_condition) | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |     # if there's not enough shop-allowed items in the pool, we can encounter gen issues | 
					
						
							|  |  |  |     # 39 is the number of shop-valid items we need to guarantee | 
					
						
							|  |  |  |     if len(itempool) < 39: | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |         itempool += create_random_items(world, shop_only_filler_weights, 39 - len(itempool)) | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |         # this is so that it passes tests and gens if you have minimal locations and only one player | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |         if world.multiworld.players == 1: | 
					
						
							|  |  |  |             for location in world.multiworld.get_unfilled_locations(player): | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |                 if "Shop Item" in location.name: | 
					
						
							|  |  |  |                     location.item = create_item(player, itempool.pop()) | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |             locations_to_fill = len(world.multiworld.get_unfilled_locations(player)) | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-19 15:31:45 -05:00
										 |  |  |     itempool += create_random_items(world, filler_weights, locations_to_fill - len(itempool)) | 
					
						
							|  |  |  |     world.multiworld.itempool += [create_item(player, name) for name in itempool] | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # 110000 - 110032 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | item_table: dict[str, ItemData] = { | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     "Trap":                                 ItemData(110000, "Traps", ItemClassification.trap), | 
					
						
							|  |  |  |     "Extra Max HP":                         ItemData(110001, "Pickups", ItemClassification.useful), | 
					
						
							|  |  |  |     "Spell Refresher":                      ItemData(110002, "Pickups", ItemClassification.filler), | 
					
						
							|  |  |  |     "Potion":                               ItemData(110003, "Items", ItemClassification.filler), | 
					
						
							|  |  |  |     "Gold (200)":                           ItemData(110004, "Gold", ItemClassification.filler), | 
					
						
							|  |  |  |     "Gold (1000)":                          ItemData(110005, "Gold", ItemClassification.filler), | 
					
						
							|  |  |  |     "Wand (Tier 1)":                        ItemData(110006, "Wands", ItemClassification.useful), | 
					
						
							|  |  |  |     "Wand (Tier 2)":                        ItemData(110007, "Wands", ItemClassification.useful), | 
					
						
							|  |  |  |     "Wand (Tier 3)":                        ItemData(110008, "Wands", ItemClassification.useful), | 
					
						
							|  |  |  |     "Wand (Tier 4)":                        ItemData(110009, "Wands", ItemClassification.useful), | 
					
						
							| 
									
										
										
										
											2023-10-28 16:27:57 -04:00
										 |  |  |     "Wand (Tier 5)":                        ItemData(110010, "Wands", ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Wand (Tier 6)":                        ItemData(110011, "Wands", ItemClassification.useful, 1), | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     "Kantele":                              ItemData(110012, "Wands", ItemClassification.useful), | 
					
						
							| 
									
										
										
										
											2024-10-16 21:31:53 -04:00
										 |  |  |     "Fire Immunity Perk":                   ItemData(110013, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Toxic Immunity Perk":                  ItemData(110014, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Explosion Immunity Perk":              ItemData(110015, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Melee Immunity Perk":                  ItemData(110016, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Electricity Immunity Perk":            ItemData(110017, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "Tinker with Wands Everywhere Perk":    ItemData(110018, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							|  |  |  |     "All-Seeing Eye Perk":                  ItemData(110019, "Perks", ItemClassification.progression | ItemClassification.useful, 1), | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     "Spatial Awareness Perk":               ItemData(110020, "Perks", ItemClassification.progression), | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |     "Extra Life Perk":                      ItemData(110021, "Repeatable Perks", ItemClassification.useful, 1), | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     "Orb":                                  ItemData(110022, "Orbs", ItemClassification.progression_skip_balancing), | 
					
						
							|  |  |  |     "Random Potion":                        ItemData(110023, "Items", ItemClassification.filler), | 
					
						
							|  |  |  |     "Secret Potion":                        ItemData(110024, "Items", ItemClassification.filler), | 
					
						
							|  |  |  |     "Powder Pouch":                         ItemData(110025, "Items", ItemClassification.filler), | 
					
						
							|  |  |  |     "Chaos Die":                            ItemData(110026, "Items", ItemClassification.filler), | 
					
						
							| 
									
										
										
										
											2024-12-19 20:30:41 -05:00
										 |  |  |     "Greed Die":                            ItemData(110027, "Items", ItemClassification.trap), | 
					
						
							| 
									
										
										
										
											2023-10-28 16:27:57 -04:00
										 |  |  |     "Kammi":                                ItemData(110028, "Items", ItemClassification.filler, 1), | 
					
						
							|  |  |  |     "Refreshing Gourd":                     ItemData(110029, "Items", ItemClassification.filler, 1), | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  |     "Sädekivi":                             ItemData(110030, "Items", ItemClassification.filler), | 
					
						
							|  |  |  |     "Broken Wand":                          ItemData(110031, "Items", ItemClassification.filler), | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | shop_only_filler_weights: dict[str, int] = { | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |     "Trap":             15, | 
					
						
							|  |  |  |     "Extra Max HP":     25, | 
					
						
							|  |  |  |     "Spell Refresher":  20, | 
					
						
							|  |  |  |     "Wand (Tier 1)":    10, | 
					
						
							|  |  |  |     "Wand (Tier 2)":    8, | 
					
						
							|  |  |  |     "Wand (Tier 3)":    7, | 
					
						
							|  |  |  |     "Wand (Tier 4)":    6, | 
					
						
							|  |  |  |     "Wand (Tier 5)":    5, | 
					
						
							|  |  |  |     "Wand (Tier 6)":    4, | 
					
						
							|  |  |  |     "Extra Life Perk":  10, | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | filler_weights: dict[str, int] = { | 
					
						
							| 
									
										
										
										
											2023-10-29 15:02:53 -04:00
										 |  |  |     **shop_only_filler_weights, | 
					
						
							|  |  |  |     "Gold (200)":       15, | 
					
						
							|  |  |  |     "Gold (1000)":      6, | 
					
						
							|  |  |  |     "Potion":           40, | 
					
						
							|  |  |  |     "Random Potion":    9, | 
					
						
							|  |  |  |     "Secret Potion":    10, | 
					
						
							|  |  |  |     "Powder Pouch":     10, | 
					
						
							|  |  |  |     "Chaos Die":        4, | 
					
						
							|  |  |  |     "Greed Die":        4, | 
					
						
							|  |  |  |     "Kammi":            4, | 
					
						
							|  |  |  |     "Refreshing Gourd": 4, | 
					
						
							|  |  |  |     "Sädekivi":         3, | 
					
						
							|  |  |  |     "Broken Wand":      10, | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | filler_items: list[str] = list(filter(lambda item: item_table[item].classification == ItemClassification.filler, | 
					
						
							|  |  |  |                                       item_table.keys())) | 
					
						
							|  |  |  | item_name_to_id: dict[str, int] = {name: data.code for name, data in item_table.items()} | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-14 07:55:45 -04:00
										 |  |  | item_name_groups: dict[str, set[str]] = { | 
					
						
							|  |  |  |     group: set(item_names) for group, item_names in itertools.groupby(item_table, lambda item: item_table[item].group) | 
					
						
							| 
									
										
										
										
											2023-04-19 20:21:56 -07:00
										 |  |  | } |