mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	Bug Squashing
This commit is contained in:
		
							
								
								
									
										1
									
								
								worlds/hk/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								worlds/hk/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | /Resources | ||||||
							
								
								
									
										16
									
								
								worlds/hk/ExtractedData.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								worlds/hk/ExtractedData.py
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										377
									
								
								worlds/hk/Extractor.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										377
									
								
								worlds/hk/Extractor.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,377 @@ | |||||||
|  | """ | ||||||
|  | Logic Extractor designed for "Randomizer 4". | ||||||
|  | Place a Randomizer 4 compatible "Resources" folder next to this script, then run the script, to create AP data. | ||||||
|  | """ | ||||||
|  | import os | ||||||
|  | import json | ||||||
|  | import typing | ||||||
|  | import ast | ||||||
|  |  | ||||||
|  | import jinja2 | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from ast import unparse | ||||||
|  | except ImportError: | ||||||
|  |     # Py 3.8 and earlier compatibility module | ||||||
|  |     from astunparse import unparse | ||||||
|  |  | ||||||
|  | from Utils import get_text_between | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def put_digits_at_end(text: str) -> str: | ||||||
|  |     for x in range(len(text)): | ||||||
|  |         if text[0].isdigit(): | ||||||
|  |             text = text[1:] + text[0] | ||||||
|  |         else: | ||||||
|  |             break | ||||||
|  |     return text | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def hk_loads(file: str) -> typing.Any: | ||||||
|  |     with open(file) as f: | ||||||
|  |         data = f.read() | ||||||
|  |     new_data = [] | ||||||
|  |     for row in data.split("\n"): | ||||||
|  |         if not row.strip().startswith(r"//"): | ||||||
|  |             new_data.append(row) | ||||||
|  |     return json.loads("\n".join(new_data)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def hk_convert(text: str) -> str: | ||||||
|  |     parts = text.replace("(", "( ").replace(")", " )").replace(">", " > ").replace("=", "==").split() | ||||||
|  |     new_parts = [] | ||||||
|  |     for part in parts: | ||||||
|  |         part = put_digits_at_end(part) | ||||||
|  |  | ||||||
|  |         if part in items or part in effect_names or part in event_names or part in connectors: | ||||||
|  |             new_parts.append(f"\"{part}\"") | ||||||
|  |         else: | ||||||
|  |             new_parts.append(part) | ||||||
|  |     text = " ".join(new_parts) | ||||||
|  |     result = "" | ||||||
|  |     parts = text.split("$StartLocation[") | ||||||
|  |     for i, part in enumerate(parts[:-1]): | ||||||
|  |         result += part + "StartLocation[\"" | ||||||
|  |         parts[i+1] = parts[i+1].replace("]", "\"]", 1) | ||||||
|  |  | ||||||
|  |     text = result + parts[-1] | ||||||
|  |  | ||||||
|  |     result = "" | ||||||
|  |     parts = text.split("COMBAT[") | ||||||
|  |     for i, part in enumerate(parts[:-1]): | ||||||
|  |         result += part + "COMBAT[\"" | ||||||
|  |         parts[i+1] = parts[i+1].replace("]", "\"]", 1) | ||||||
|  |  | ||||||
|  |     text = result + parts[-1] | ||||||
|  |     return text.replace("+", "and").replace("|", "or").replace("$", "").strip() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Absorber(ast.NodeTransformer): | ||||||
|  |     additional_truths = set() | ||||||
|  |     additional_falses = set() | ||||||
|  |  | ||||||
|  |     def __init__(self, truth_values, false_values): | ||||||
|  |         self.truth_values = truth_values | ||||||
|  |         self.truth_values |= {"True", "None", "ANY", "ITEMRANDO"} | ||||||
|  |         self.false_values = false_values | ||||||
|  |         self.false_values |= {"False", "NONE", "RANDOMELEVATORS"} | ||||||
|  |  | ||||||
|  |         super(Absorber, self).__init__() | ||||||
|  |  | ||||||
|  |     def generic_visit(self, node: ast.AST) -> ast.AST: | ||||||
|  |         # Need to call super() in any case to visit child nodes of the current one. | ||||||
|  |         node = super().generic_visit(node) | ||||||
|  |         return node | ||||||
|  |  | ||||||
|  |     def visit_BoolOp(self, node: ast.BoolOp) -> ast.AST: | ||||||
|  |         if type(node.op) == ast.And: | ||||||
|  |             if self.is_always_true(node.values[0]): | ||||||
|  |                 return self.visit(node.values[1]) | ||||||
|  |             if self.is_always_true(node.values[1]): | ||||||
|  |                 return self.visit(node.values[0]) | ||||||
|  |             if self.is_always_false(node.values[0]) or self.is_always_false(node.values[1]): | ||||||
|  |                 return ast.Constant(False, ctx=ast.Load()) | ||||||
|  |         elif type(node.op) == ast.Or: | ||||||
|  |             if self.is_always_true(node.values[0]) or self.is_always_true(node.values[1]): | ||||||
|  |                 return ast.Constant(True, ctx=ast.Load()) | ||||||
|  |             if self.is_always_false(node.values[0]): | ||||||
|  |                 return self.visit(node.values[1]) | ||||||
|  |             if self.is_always_false(node.values[1]): | ||||||
|  |                 return self.visit(node.values[0]) | ||||||
|  |         return self.generic_visit(node) | ||||||
|  |  | ||||||
|  |     def visit_Name(self, node: ast.Name) -> ast.AST: | ||||||
|  |         if node.id in self.truth_values: | ||||||
|  |             return ast.Constant(True, ctx=node.ctx) | ||||||
|  |         if node.id in self.false_values: | ||||||
|  |             return ast.Constant(False, ctx=node.ctx) | ||||||
|  |         if node.id in logic_options: | ||||||
|  |             return ast.Call( | ||||||
|  |                 func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='_kh_option', ctx=ast.Load()), | ||||||
|  |                 args=[ast.Name(id="player", ctx=ast.Load()), ast.Constant(value=logic_options[node.id])], keywords=[]) | ||||||
|  |         if node.id in macros: | ||||||
|  |             return macros[node.id].body | ||||||
|  |         if node.id in region_names: | ||||||
|  |             raise Exception(f"Should be event {node.id}") | ||||||
|  |         # You'd think this means reach Scene/Region of that name, but is actually waypoint/event | ||||||
|  |         # if node.id in region_names: | ||||||
|  |         #     return ast.Call( | ||||||
|  |         #         func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='can_reach', ctx=ast.Load()), | ||||||
|  |         #         args=[ast.Constant(value=node.id), | ||||||
|  |         #               ast.Constant(value="Region"), | ||||||
|  |         #               ast.Name(id="player", ctx=ast.Load())], | ||||||
|  |         #         keywords=[]) | ||||||
|  |         return self.generic_visit(node) | ||||||
|  |  | ||||||
|  |     def visit_Constant(self, node: ast.Constant) -> ast.AST: | ||||||
|  |         if type(node.value) == str: | ||||||
|  |             logic_items.add(node.value) | ||||||
|  |             return ast.Call( | ||||||
|  |                 func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='count', ctx=ast.Load()), | ||||||
|  |                 args=[ast.Constant(value=node.value), ast.Name(id="player", ctx=ast.Load())], keywords=[]) | ||||||
|  |  | ||||||
|  |         return node | ||||||
|  |  | ||||||
|  |     def visit_Subscript(self, node: ast.Subscript) -> ast.AST: | ||||||
|  |         if node.value.id == "NotchCost": | ||||||
|  |             notches = [ast.Constant(value=notch.value - 1) for notch in node.slice.elts]  # apparently 1-indexed | ||||||
|  |             return ast.Call( | ||||||
|  |                 func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='_hk_notches', ctx=ast.Load()), | ||||||
|  |                 args=[ast.Name(id="player", ctx=ast.Load())] + notches, keywords=[]) | ||||||
|  |         elif node.value.id == "StartLocation": | ||||||
|  |             node.slice.value = node.slice.value.replace(" ", "_").lower() | ||||||
|  |             if node.slice.value in removed_starts: | ||||||
|  |                 return ast.Constant(False, ctx=node.ctx) | ||||||
|  |             return ast.Call( | ||||||
|  |                 func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='_kh_start', ctx=ast.Load()), | ||||||
|  |                 args=[ast.Name(id="player", ctx=ast.Load()), node.slice], keywords=[]) | ||||||
|  |         elif node.value.id == "COMBAT": | ||||||
|  |             return macros[unparse(node)].body | ||||||
|  |         else: | ||||||
|  |             name = unparse(node) | ||||||
|  |             if name in self.additional_truths: | ||||||
|  |                 return ast.Constant(True, ctx=ast.Load()) | ||||||
|  |             elif name in self.additional_falses: | ||||||
|  |                 return ast.Constant(False, ctx=ast.Load()) | ||||||
|  |             elif name in macros: | ||||||
|  |                 # macro such as "COMBAT[White_Palace_Arenas]" | ||||||
|  |                 return macros[name].body | ||||||
|  |             else: | ||||||
|  |                 # assume Entrance | ||||||
|  |                 entrance = unparse(node) | ||||||
|  |                 assert entrance in connectors, entrance | ||||||
|  |                 return ast.Call( | ||||||
|  |                     func=ast.Attribute(value=ast.Name(id='state', ctx=ast.Load()), attr='can_reach', ctx=ast.Load()), | ||||||
|  |                     args=[ast.Constant(value=entrance), | ||||||
|  |                           ast.Constant(value="Entrance"), | ||||||
|  |                           ast.Name(id="player", ctx=ast.Load())], | ||||||
|  |                     keywords=[]) | ||||||
|  |         return node | ||||||
|  |  | ||||||
|  |     def is_always_true(self, node): | ||||||
|  |         if isinstance(node, ast.Name) and (node.id in self.truth_values or node.id in self.additional_truths): | ||||||
|  |             return True | ||||||
|  |         if isinstance(node, ast.Subscript) and unparse(node) in self.additional_truths: | ||||||
|  |             return True | ||||||
|  |  | ||||||
|  |     def is_always_false(self, node): | ||||||
|  |         if isinstance(node, ast.Name) and (node.id in self.false_values or node.id in self.additional_falses): | ||||||
|  |             return True | ||||||
|  |         if isinstance(node, ast.Subscript) and unparse(node) in self.additional_falses: | ||||||
|  |             return True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_parser(truths: typing.Set[str] = frozenset(), falses: typing.Set[str] = frozenset()): | ||||||
|  |     return Absorber(truths, falses) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def ast_parse(parser, rule_text, truths: typing.Set[str] = frozenset(), falses: typing.Set[str] = frozenset()): | ||||||
|  |     tree = ast.parse(hk_convert(rule_text), mode='eval') | ||||||
|  |     parser.additional_truths = truths | ||||||
|  |     parser.additional_falses = falses | ||||||
|  |     new_tree = parser.visit(tree) | ||||||
|  |     parser.additional_truths = set() | ||||||
|  |     parser.additional_truths = set() | ||||||
|  |     return new_tree | ||||||
|  |  | ||||||
|  |  | ||||||
|  | world_folder = os.path.dirname(__file__) | ||||||
|  |  | ||||||
|  | resources_source = os.path.join(world_folder, "Resources") | ||||||
|  | data_folder = os.path.join(resources_source, "Data") | ||||||
|  | logic_folder = os.path.join(resources_source, "Logic") | ||||||
|  | logic_options: typing.Dict[str, str] = hk_loads(os.path.join(data_folder, "logic_settings.json")) | ||||||
|  | for logic_key, logic_value in logic_options.items(): | ||||||
|  |     logic_options[logic_key] = logic_value.split(".", 1)[-1] | ||||||
|  | del (logic_options["RANDOMELEVATORS"]) | ||||||
|  | extra_pool_options: typing.List[typing.Dict[str, typing.Any]] = hk_loads(os.path.join(data_folder, "pools.json")) | ||||||
|  | pool_options: typing.Dict[str, typing.Tuple[typing.List[str], typing.List[str]]] = {} | ||||||
|  | for option in extra_pool_options: | ||||||
|  |     if option["Path"] != "False": | ||||||
|  |         items: typing.List[str] = [] | ||||||
|  |         locations: typing.List[str] = [] | ||||||
|  |         for pairing in option["Vanilla"]: | ||||||
|  |             items.append(pairing["item"]) | ||||||
|  |             location_name = pairing["location"] | ||||||
|  |             if any(cost_entry["term"] == "CHARMS" for cost_entry in pairing.get("costs", [])): | ||||||
|  |                 location_name += "_(Requires_Charms)" | ||||||
|  |             locations.append(location_name) | ||||||
|  |         if option["Path"]: | ||||||
|  |             # basename carries over from prior entry if no Path given | ||||||
|  |             basename = option["Path"].split(".", 1)[-1] | ||||||
|  |         if not basename.startswith("Randomize"): | ||||||
|  |             basename = "Randomize" + basename | ||||||
|  |         assert len(items) == len(locations) | ||||||
|  |         if items:  # skip empty pools | ||||||
|  |             if basename in pool_options: | ||||||
|  |                 pool_options[basename] = pool_options[basename][0]+items, pool_options[basename][1]+locations | ||||||
|  |             else: | ||||||
|  |                 pool_options[basename] = items, locations | ||||||
|  | del extra_pool_options | ||||||
|  |  | ||||||
|  | # items | ||||||
|  | items: typing.Dict[str, typing.Dict] = hk_loads(os.path.join(data_folder, "items.json")) | ||||||
|  | logic_items: typing.Set[str] = set() | ||||||
|  | for item_name in sorted(items): | ||||||
|  |     item = items[item_name] | ||||||
|  |     items[item_name] = item["Pool"] | ||||||
|  | items: typing.Dict[str, str] | ||||||
|  |  | ||||||
|  | extra_item_data: typing.List[typing.Dict[str, typing.Any]] = hk_loads(os.path.join(logic_folder, "items.json")) | ||||||
|  | item_effects: typing.Dict[str, typing.Dict[str, int]] = {} | ||||||
|  | effect_names: typing.Set[str] = set() | ||||||
|  | for item_data in extra_item_data: | ||||||
|  |     if "FalseItem" in item_data: | ||||||
|  |         item_data = item_data["FalseItem"] | ||||||
|  |     effects = [] | ||||||
|  |     if "Effect" in item_data: | ||||||
|  |         effects = [item_data["Effect"]] | ||||||
|  |     elif "Effects" in item_data: | ||||||
|  |         effects = item_data["Effects"] | ||||||
|  |     for effect in effects: | ||||||
|  |         effect_names.add(effect["Term"]) | ||||||
|  |     effects = {effect["Term"]: effect["Value"] for effect in effects if | ||||||
|  |                effect["Term"] != item_data["Name"] and effect["Term"] != "GEO"} | ||||||
|  |     if effects: | ||||||
|  |         item_effects[item_data["Name"]] = effects | ||||||
|  |  | ||||||
|  | del extra_item_data | ||||||
|  |  | ||||||
|  | # locations | ||||||
|  | original_locations: typing.Dict[str, typing.Dict[str, typing.Any]] = hk_loads(os.path.join(data_folder, "locations.json")) | ||||||
|  | del(original_locations["Start"])  # Starting Inventory works different in AP | ||||||
|  |  | ||||||
|  | locations: typing.List[str] = [] | ||||||
|  | locations_in_regions: typing.Dict[str, typing.List[str]] = {} | ||||||
|  | location_to_region_lookup: typing.Dict[str, str] = {} | ||||||
|  | multi_locations: typing.Dict[str, typing.List[str]] = {} | ||||||
|  | for location_name, location_data in original_locations.items(): | ||||||
|  |     region_name = location_data["SceneName"] | ||||||
|  |     if location_data["FlexibleCount"]: | ||||||
|  |         location_names = [f"{location_name}_{count}" for count in range(1, 17)] | ||||||
|  |         multi_locations[location_name] = location_names | ||||||
|  |     else: | ||||||
|  |         location_names = [location_name] | ||||||
|  |  | ||||||
|  |     location_to_region_lookup.update({name: region_name for name in location_names}) | ||||||
|  |     locations_in_regions.setdefault(region_name, []).extend(location_names) | ||||||
|  |     locations.extend(location_names) | ||||||
|  | del original_locations | ||||||
|  |  | ||||||
|  | # regions | ||||||
|  | region_names: typing.Set[str] = set(hk_loads(os.path.join(data_folder, "rooms.json"))) | ||||||
|  | connectors_data: typing.Dict[str, typing.Dict[str, typing.Any]] = hk_loads(os.path.join(data_folder, "transitions.json")) | ||||||
|  | connectors_logic: typing.List[typing.Dict[str, typing.Any]] = hk_loads(os.path.join(logic_folder, "transitions.json")) | ||||||
|  | exits: typing.Dict[str, typing.List[str]] = {} | ||||||
|  | connectors: typing.Dict[str, str] = {} | ||||||
|  | one_ways: typing.Set[str] = set() | ||||||
|  | for connector_name, connector_data in connectors_data.items(): | ||||||
|  |     exits.setdefault(connector_data["SceneName"], []).append(connector_name) | ||||||
|  |     connectors[connector_name] = connector_data["VanillaTarget"] | ||||||
|  |     if connector_data["Sides"] != "Both": | ||||||
|  |         one_ways.add(connector_name) | ||||||
|  | del connectors_data | ||||||
|  |  | ||||||
|  | # starts | ||||||
|  | starts: typing.Dict[str, typing.Dict[str, typing.Any]] = hk_loads(os.path.join(data_folder, "starts.json")) | ||||||
|  |  | ||||||
|  | # only allow always valid starts for now | ||||||
|  | removed_starts: typing.Set[str] = {name.replace(" ", "_").lower() for name, data in starts.items() if | ||||||
|  |                                    name != "King's Pass"} | ||||||
|  |  | ||||||
|  | starts: typing.Dict[str, str] = { | ||||||
|  |     name.replace(" ", "_").lower(): data["sceneName"] for name, data in starts.items() if name == "King's Pass"} | ||||||
|  |  | ||||||
|  | # logic | ||||||
|  | falses = {"MAPAREARANDO", "FULLAREARANDO"} | ||||||
|  | macros: typing.Dict[str, ast.AST] = { | ||||||
|  | } | ||||||
|  | parser = get_parser(set(), falses) | ||||||
|  | extra_macros: typing.Dict[str, str] = hk_loads(os.path.join(logic_folder, "macros.json")) | ||||||
|  | raw_location_rules: typing.List[typing.Dict[str, str]] = hk_loads(os.path.join(logic_folder, "locations.json")) | ||||||
|  | events: typing.List[typing.Dict[str, typing.Any]] = hk_loads(os.path.join(logic_folder, "waypoints.json")) | ||||||
|  |  | ||||||
|  | event_names: typing.Set[str] = {event["name"] for event in events} | ||||||
|  |  | ||||||
|  | for macro_name, rule in extra_macros.items(): | ||||||
|  |     if macro_name not in macros: | ||||||
|  |         macro_name = put_digits_at_end(macro_name) | ||||||
|  |         if macro_name in items or macro_name in effect_names: | ||||||
|  |             continue | ||||||
|  |         assert macro_name not in events | ||||||
|  |         rule = ast_parse(parser, rule) | ||||||
|  |         macros[macro_name] = rule | ||||||
|  |         if macro_name.startswith("COMBAT["): | ||||||
|  |             name = get_text_between(macro_name, "COMBAT[", "]") | ||||||
|  |             if not "'" in name: | ||||||
|  |                 macros[f"COMBAT['{name}']"] = rule | ||||||
|  |             macros[f'COMBAT["{name}"]'] = rule | ||||||
|  |  | ||||||
|  | location_rules: typing.Dict[str, str] = {} | ||||||
|  | for loc_obj in raw_location_rules: | ||||||
|  |     loc_name = loc_obj["name"] | ||||||
|  |     rule = loc_obj["logic"] | ||||||
|  |     if rule != "ANY": | ||||||
|  |         rule = ast_parse(parser, rule) | ||||||
|  |         location_rules[loc_name] = unparse(rule) | ||||||
|  | location_rules["Salubra_(Requires_Charms)"] = location_rules["Salubra"] | ||||||
|  |  | ||||||
|  | connectors_rules: typing.Dict[str, str] = {} | ||||||
|  | for connector_obj in connectors_logic: | ||||||
|  |     name = connector_obj["Name"] | ||||||
|  |     rule = connector_obj["logic"] | ||||||
|  |     rule = ast_parse(parser, rule) | ||||||
|  |     rule = unparse(rule) | ||||||
|  |     if rule != "True": | ||||||
|  |         connectors_rules[name] = rule | ||||||
|  |  | ||||||
|  | event_rules: typing.Dict[str, str] = {} | ||||||
|  | for event in events: | ||||||
|  |     rule = ast_parse(parser, event["logic"]) | ||||||
|  |     rule = unparse(rule) | ||||||
|  |     if rule != "True": | ||||||
|  |         event_rules[event["name"]] = rule | ||||||
|  |  | ||||||
|  |  | ||||||
|  | event_rules.update(connectors_rules) | ||||||
|  | connectors_rules = {} | ||||||
|  |  | ||||||
|  | names = sorted({"logic_options", "starts", "pool_options", "locations", "multi_locations", "location_to_region_lookup", | ||||||
|  |                 "event_names", "item_effects", "items", "logic_items", "region_names", | ||||||
|  |                 "exits", "connectors", "one_ways"}) | ||||||
|  | warning = "# This module is written by Extractor.py, do not edit manually!.\n\n" | ||||||
|  | with open(os.path.join(os.path.dirname(__file__), "ExtractedData.py"), "wt") as py: | ||||||
|  |     py.write(warning) | ||||||
|  |     for name in names: | ||||||
|  |         py.write(f"{name} = {globals()[name]}\n") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | template_env: jinja2.Environment = \ | ||||||
|  |     jinja2.Environment(loader=jinja2.FileSystemLoader([os.path.join(os.path.dirname(__file__), "templates")])) | ||||||
|  | rules_template = template_env.get_template("RulesTemplate.pyt") | ||||||
|  | rules = rules_template.render(location_rules=location_rules, one_ways=one_ways, connectors_rules=connectors_rules, | ||||||
|  |                               event_rules=event_rules) | ||||||
|  |  | ||||||
|  | with open("Rules.py", "wt") as py: | ||||||
|  |     py.write(warning) | ||||||
|  |     py.write(rules) | ||||||
| @@ -1,397 +1,20 @@ | |||||||
| # generated by https://github.com/Berserker66/HollowKnight.RandomizerMod/blob/master/extract_data.py | from typing import Dict, Set, NamedTuple | ||||||
| # do not edit manually | from .ExtractedData import items, logic_items, item_effects | ||||||
|  |  | ||||||
| from .Types import HKItemData | item_table = {} | ||||||
| from typing import Dict, Set |  | ||||||
|  |  | ||||||
| item_table = \ |  | ||||||
| {   '150_Geo-Resting_Grounds_Chest': HKItemData(advancement=False, id=16777336, type='Geo'), |  | ||||||
|     '160_Geo-Weavers_Den_Chest': HKItemData(advancement=False, id=16777338, type='Geo'), |  | ||||||
|     '1_Geo': HKItemData(advancement=False, id=16777339, type='Fake'), |  | ||||||
|     '200_Geo-False_Knight_Chest': HKItemData(advancement=False, id=16777331, type='Geo'), |  | ||||||
|     '380_Geo-Soul_Master_Chest': HKItemData(advancement=False, id=16777332, type='Geo'), |  | ||||||
|     '620_Geo-Mantis_Lords_Chest': HKItemData(advancement=False, id=16777335, type='Geo'), |  | ||||||
|     '655_Geo-Watcher_Knights_Chest': HKItemData(advancement=False, id=16777333, type='Geo'), |  | ||||||
|     '80_Geo-Crystal_Peak_Chest': HKItemData(advancement=False, id=16777337, type='Geo'), |  | ||||||
|     '85_Geo-Greenpath_Chest': HKItemData(advancement=False, id=16777334, type='Geo'), |  | ||||||
|     'Abyss': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Abyss_Shriek': HKItemData(advancement=True, id=16777236, type='Skill'), |  | ||||||
|     'Ancestral_Mound': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Ancient_Basin_Map': HKItemData(advancement=False, id=16777482, type='Map'), |  | ||||||
|     'Arcane_Egg-Birthplace': HKItemData(advancement=False, id=16777402, type='Relic'), |  | ||||||
|     'Arcane_Egg-Lifeblood_Core': HKItemData(advancement=False, id=16777400, type='Relic'), |  | ||||||
|     'Arcane_Egg-Seer': HKItemData(advancement=False, id=16777399, type='Relic'), |  | ||||||
|     'Arcane_Egg-Shade_Cloak': HKItemData(advancement=False, id=16777401, type='Relic'), |  | ||||||
|     'Awoken_Dream_Nail': HKItemData(advancement=True, id=16777230, type='Skill'), |  | ||||||
|     'Baldur_Shell': HKItemData(advancement=False, id=16777245, type='Charm'), |  | ||||||
|     "Beast's_Den": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Blue_Lake': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Boss_Essence-Elder_Hu': HKItemData(advancement=True, id=16777418, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Failed_Champion': HKItemData(advancement=True, id=16777425, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Galien': HKItemData(advancement=True, id=16777423, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Gorb': HKItemData(advancement=True, id=16777420, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Grey_Prince_Zote': HKItemData(advancement=True, id=16777429, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Lost_Kin': HKItemData(advancement=True, id=16777427, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Markoth': HKItemData(advancement=True, id=16777424, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Marmu': HKItemData(advancement=True, id=16777421, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-No_Eyes': HKItemData(advancement=True, id=16777422, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Soul_Tyrant': HKItemData(advancement=True, id=16777426, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-White_Defender': HKItemData(advancement=True, id=16777428, type='Essence_Boss'), |  | ||||||
|     'Boss_Essence-Xero': HKItemData(advancement=True, id=16777419, type='Essence_Boss'), |  | ||||||
|     'Bottom_Left_Fungal_Wastes': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Bottom_Left_Queen's_Gardens": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Bottom_Right_Queen's_Gardens": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Can_Stag': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Cast_Off_Shell': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Center_Right_Kingdom's_Edge": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Central_Kingdom's_Edge": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Central_Left_Waterways': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Charm_Notch-Colosseum': HKItemData(advancement=False, id=16777323, type='Notch'), |  | ||||||
|     'Charm_Notch-Fog_Canyon': HKItemData(advancement=False, id=16777322, type='Notch'), |  | ||||||
|     'Charm_Notch-Grimm': HKItemData(advancement=False, id=16777324, type='Notch'), |  | ||||||
|     'Charm_Notch-Shrumal_Ogres': HKItemData(advancement=False, id=16777321, type='Notch'), |  | ||||||
|     'City_Crest': HKItemData(advancement=True, id=16777283, type='Key'), |  | ||||||
|     'City_Storerooms_Stag': HKItemData(advancement=True, id=16777495, type='Stag'), |  | ||||||
|     'City_of_Tears_Map': HKItemData(advancement=False, id=16777484, type='Map'), |  | ||||||
|     "Collector's_Map": HKItemData(advancement=False, id=16777295, type='Key'), |  | ||||||
|     'Colosseum': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Crossroads': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Crossroads_Map': HKItemData(advancement=False, id=16777476, type='Map'), |  | ||||||
|     'Crossroads_Stag': HKItemData(advancement=True, id=16777491, type='Stag'), |  | ||||||
|     'Crystal_Heart': HKItemData(advancement=True, id=16777224, type='Skill'), |  | ||||||
|     'Crystal_Peak': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Crystal_Peak_Map': HKItemData(advancement=False, id=16777487, type='Map'), |  | ||||||
|     'Crystallized_Mound': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Cyclone_Slash': HKItemData(advancement=True, id=16777237, type='Skill'), |  | ||||||
|     'Dark_Deepnest': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Dash_Slash': HKItemData(advancement=True, id=16777238, type='Skill'), |  | ||||||
|     'Dashmaster': HKItemData(advancement=True, id=16777271, type='Charm'), |  | ||||||
|     'Deep_Focus': HKItemData(advancement=False, id=16777274, type='Charm'), |  | ||||||
|     'Deepnest': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Deepnest_Map-Right_[Gives_Quill]': HKItemData(advancement=False, id=16777481, type='Map'), |  | ||||||
|     'Deepnest_Map-Upper': HKItemData(advancement=False, id=16777480, type='Map'), |  | ||||||
|     "Defender's_Crest": HKItemData(advancement=False, id=16777250, type='Charm'), |  | ||||||
|     'Descending_Dark': HKItemData(advancement=True, id=16777234, type='Skill'), |  | ||||||
|     'Desolate_Dive': HKItemData(advancement=True, id=16777233, type='Skill'), |  | ||||||
|     'Dirtmouth': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Dirtmouth_Stag': HKItemData(advancement=True, id=16777490, type='Stag'), |  | ||||||
|     'Distant_Village': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Distant_Village_Stag': HKItemData(advancement=True, id=16777498, type='Stag'), |  | ||||||
|     'Dream_Gate': HKItemData(advancement=True, id=16777229, type='Skill'), |  | ||||||
|     'Dream_Nail': HKItemData(advancement=True, id=16777228, type='Skill'), |  | ||||||
|     'Dream_Wielder': HKItemData(advancement=False, id=16777270, type='Charm'), |  | ||||||
|     'Dreamer': HKItemData(advancement=True, id=16777221, type='Fake'), |  | ||||||
|     'Dreamshield': HKItemData(advancement=False, id=16777280, type='Charm'), |  | ||||||
|     'Elegant_Key': HKItemData(advancement=True, id=16777291, type='Key'), |  | ||||||
|     'Emilitia': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Equipped': HKItemData(advancement=False, id=16777521, type='Fake'), |  | ||||||
|     'Failed_Tramway': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Far_Left_Basin': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Far_Left_Waterways': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Far_Queen's_Gardens": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Far_Right_Deepnest': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Flukenest': HKItemData(advancement=False, id=16777251, type='Charm'), |  | ||||||
|     'Focus': HKItemData(advancement=True, id=16777240, type='Cursed'), |  | ||||||
|     'Fog_Canyon_Map': HKItemData(advancement=False, id=16777478, type='Map'), |  | ||||||
|     'Fragile_Greed': HKItemData(advancement=False, id=16777264, type='Charm'), |  | ||||||
|     'Fragile_Heart': HKItemData(advancement=False, id=16777263, type='Charm'), |  | ||||||
|     'Fragile_Strength': HKItemData(advancement=False, id=16777265, type='Charm'), |  | ||||||
|     'Fungal_Core': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Fungal_Wastes': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Fungal_Wastes_Map': HKItemData(advancement=False, id=16777479, type='Map'), |  | ||||||
|     'Fury_of_the_Fallen': HKItemData(advancement=False, id=16777246, type='Charm'), |  | ||||||
|     'Gathering_Swarm': HKItemData(advancement=False, id=16777241, type='Charm'), |  | ||||||
|     'Glowing_Womb': HKItemData(advancement=True, id=16777262, type='Charm'), |  | ||||||
|     'Godtuner': HKItemData(advancement=False, id=16777294, type='Key'), |  | ||||||
|     'Great_Slash': HKItemData(advancement=True, id=16777239, type='Skill'), |  | ||||||
|     'Greenpath': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Greenpath-QG': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Greenpath_Map': HKItemData(advancement=False, id=16777477, type='Map'), |  | ||||||
|     'Greenpath_Stag': HKItemData(advancement=True, id=16777492, type='Stag'), |  | ||||||
|     'Grimmchild': HKItemData(advancement=True, id=16777282, type='Charm'), |  | ||||||
|     'Grimmkin_Flame-Ancient_Basin': HKItemData(advancement=True, id=16777516, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Brumm': HKItemData(advancement=True, id=16777518, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-City_Storerooms': HKItemData(advancement=True, id=16777509, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Crystal_Peak': HKItemData(advancement=True, id=16777511, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Fungal_Core': HKItemData(advancement=True, id=16777515, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Greenpath': HKItemData(advancement=True, id=16777510, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Hive': HKItemData(advancement=True, id=16777517, type='Flame'), |  | ||||||
|     "Grimmkin_Flame-King's_Pass": HKItemData(advancement=True, id=16777512, type='Flame'), |  | ||||||
|     "Grimmkin_Flame-Kingdom's_Edge": HKItemData(advancement=True, id=16777514, type='Flame'), |  | ||||||
|     'Grimmkin_Flame-Resting_Grounds': HKItemData(advancement=True, id=16777513, type='Flame'), |  | ||||||
|     'Grub-Basin_Requires_Dive': HKItemData(advancement=True, id=16777452, type='Grub'), |  | ||||||
|     'Grub-Basin_Requires_Wings': HKItemData(advancement=True, id=16777451, type='Grub'), |  | ||||||
|     "Grub-Beast's_Den": HKItemData(advancement=True, id=16777446, type='Grub'), |  | ||||||
|     'Grub-City_of_Tears_Guarded': HKItemData(advancement=True, id=16777459, type='Grub'), |  | ||||||
|     'Grub-City_of_Tears_Left': HKItemData(advancement=True, id=16777456, type='Grub'), |  | ||||||
|     'Grub-Collector_1': HKItemData(advancement=True, id=16777473, type='Grub'), |  | ||||||
|     'Grub-Collector_2': HKItemData(advancement=True, id=16777474, type='Grub'), |  | ||||||
|     'Grub-Collector_3': HKItemData(advancement=True, id=16777475, type='Grub'), |  | ||||||
|     'Grub-Crossroads_Acid': HKItemData(advancement=True, id=16777430, type='Grub'), |  | ||||||
|     'Grub-Crossroads_Center': HKItemData(advancement=True, id=16777431, type='Grub'), |  | ||||||
|     'Grub-Crossroads_Guarded': HKItemData(advancement=True, id=16777434, type='Grub'), |  | ||||||
|     'Grub-Crossroads_Spike': HKItemData(advancement=True, id=16777433, type='Grub'), |  | ||||||
|     'Grub-Crossroads_Stag': HKItemData(advancement=True, id=16777432, type='Grub'), |  | ||||||
|     'Grub-Crystal_Heart': HKItemData(advancement=True, id=16777467, type='Grub'), |  | ||||||
|     'Grub-Crystal_Peak_Below_Chest': HKItemData(advancement=True, id=16777462, type='Grub'), |  | ||||||
|     'Grub-Crystal_Peak_Crushers': HKItemData(advancement=True, id=16777466, type='Grub'), |  | ||||||
|     'Grub-Crystal_Peak_Mimic': HKItemData(advancement=True, id=16777465, type='Grub'), |  | ||||||
|     'Grub-Crystal_Peak_Spike': HKItemData(advancement=True, id=16777464, type='Grub'), |  | ||||||
|     'Grub-Crystallized_Mound': HKItemData(advancement=True, id=16777463, type='Grub'), |  | ||||||
|     'Grub-Dark_Deepnest': HKItemData(advancement=True, id=16777445, type='Grub'), |  | ||||||
|     'Grub-Deepnest_Mimic': HKItemData(advancement=True, id=16777442, type='Grub'), |  | ||||||
|     'Grub-Deepnest_Nosk': HKItemData(advancement=True, id=16777443, type='Grub'), |  | ||||||
|     'Grub-Deepnest_Spike': HKItemData(advancement=True, id=16777444, type='Grub'), |  | ||||||
|     'Grub-Fog_Canyon': HKItemData(advancement=True, id=16777439, type='Grub'), |  | ||||||
|     'Grub-Fungal_Bouncy': HKItemData(advancement=True, id=16777440, type='Grub'), |  | ||||||
|     'Grub-Fungal_Spore_Shroom': HKItemData(advancement=True, id=16777441, type='Grub'), |  | ||||||
|     'Grub-Greenpath_Cornifer': HKItemData(advancement=True, id=16777435, type='Grub'), |  | ||||||
|     'Grub-Greenpath_Journal': HKItemData(advancement=True, id=16777436, type='Grub'), |  | ||||||
|     'Grub-Greenpath_MMC': HKItemData(advancement=True, id=16777437, type='Grub'), |  | ||||||
|     'Grub-Greenpath_Stag': HKItemData(advancement=True, id=16777438, type='Grub'), |  | ||||||
|     'Grub-Hallownest_Crown': HKItemData(advancement=True, id=16777468, type='Grub'), |  | ||||||
|     'Grub-Hive_External': HKItemData(advancement=True, id=16777449, type='Grub'), |  | ||||||
|     'Grub-Hive_Internal': HKItemData(advancement=True, id=16777450, type='Grub'), |  | ||||||
|     'Grub-Howling_Cliffs': HKItemData(advancement=True, id=16777469, type='Grub'), |  | ||||||
|     "Grub-King's_Station": HKItemData(advancement=True, id=16777460, type='Grub'), |  | ||||||
|     "Grub-Kingdom's_Edge_Camp": HKItemData(advancement=True, id=16777448, type='Grub'), |  | ||||||
|     "Grub-Kingdom's_Edge_Oro": HKItemData(advancement=True, id=16777447, type='Grub'), |  | ||||||
|     "Grub-Queen's_Gardens_Marmu": HKItemData(advancement=True, id=16777471, type='Grub'), |  | ||||||
|     "Grub-Queen's_Gardens_Stag": HKItemData(advancement=True, id=16777470, type='Grub'), |  | ||||||
|     "Grub-Queen's_Gardens_Top": HKItemData(advancement=True, id=16777472, type='Grub'), |  | ||||||
|     'Grub-Resting_Grounds': HKItemData(advancement=True, id=16777461, type='Grub'), |  | ||||||
|     'Grub-Soul_Sanctum': HKItemData(advancement=True, id=16777457, type='Grub'), |  | ||||||
|     "Grub-Watcher's_Spire": HKItemData(advancement=True, id=16777458, type='Grub'), |  | ||||||
|     'Grub-Waterways_East': HKItemData(advancement=True, id=16777454, type='Grub'), |  | ||||||
|     'Grub-Waterways_Main': HKItemData(advancement=True, id=16777453, type='Grub'), |  | ||||||
|     'Grub-Waterways_Requires_Tram': HKItemData(advancement=True, id=16777455, type='Grub'), |  | ||||||
|     "Grubberfly's_Elegy": HKItemData(advancement=True, id=16777275, type='Charm'), |  | ||||||
|     'Grubfather': HKItemData(advancement=False, id=16777519, type='Fake'), |  | ||||||
|     'Grubsong': HKItemData(advancement=False, id=16777243, type='Charm'), |  | ||||||
|     "Hallownest's_Crown": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Hallownest_Seal-Beast's_Den": HKItemData(advancement=False, id=16777389, type='Relic'), |  | ||||||
|     'Hallownest_Seal-City_Rafters': HKItemData(advancement=False, id=16777385, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Crossroads_Well': HKItemData(advancement=False, id=16777374, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Deepnest_By_Mantis_Lords': HKItemData(advancement=False, id=16777388, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Fog_Canyon_East': HKItemData(advancement=False, id=16777378, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Fog_Canyon_West': HKItemData(advancement=False, id=16777377, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Fungal_Wastes_Sporgs': HKItemData(advancement=False, id=16777380, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Greenpath': HKItemData(advancement=False, id=16777376, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Grubs': HKItemData(advancement=False, id=16777375, type='Relic'), |  | ||||||
|     "Hallownest_Seal-King's_Station": HKItemData(advancement=False, id=16777384, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Mantis_Lords': HKItemData(advancement=False, id=16777381, type='Relic'), |  | ||||||
|     "Hallownest_Seal-Queen's_Gardens": HKItemData(advancement=False, id=16777390, type='Relic'), |  | ||||||
|     "Hallownest_Seal-Queen's_Station": HKItemData(advancement=False, id=16777379, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Resting_Grounds_Catacombs': HKItemData(advancement=False, id=16777383, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Seer': HKItemData(advancement=False, id=16777382, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Soul_Sanctum': HKItemData(advancement=False, id=16777386, type='Relic'), |  | ||||||
|     'Hallownest_Seal-Watcher_Knight': HKItemData(advancement=False, id=16777387, type='Relic'), |  | ||||||
|     'Heavy_Blow': HKItemData(advancement=False, id=16777255, type='Charm'), |  | ||||||
|     'Herrah': HKItemData(advancement=True, id=16777219, type='Dreamer'), |  | ||||||
|     'Hidden_Station_Stag': HKItemData(advancement=True, id=16777499, type='Stag'), |  | ||||||
|     'Hive': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Hiveblood': HKItemData(advancement=False, id=16777269, type='Charm'), |  | ||||||
|     'Hollow Knight': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Howling_Cliffs': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Howling_Cliffs_Map': HKItemData(advancement=False, id=16777486, type='Map'), |  | ||||||
|     'Howling_Wraiths': HKItemData(advancement=True, id=16777235, type='Skill'), |  | ||||||
|     "Isma's_Grove": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Isma's_Tear": HKItemData(advancement=True, id=16777227, type='Skill'), |  | ||||||
|     "Joni's_Blessing": HKItemData(advancement=True, id=16777267, type='Charm'), |  | ||||||
|     'Junk_Pit': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "King's_Brand": HKItemData(advancement=True, id=16777293, type='Key'), |  | ||||||
|     "King's_Idol-Cliffs": HKItemData(advancement=False, id=16777392, type='Relic'), |  | ||||||
|     "King's_Idol-Crystal_Peak": HKItemData(advancement=False, id=16777393, type='Relic'), |  | ||||||
|     "King's_Idol-Deepnest": HKItemData(advancement=False, id=16777398, type='Relic'), |  | ||||||
|     "King's_Idol-Dung_Defender": HKItemData(advancement=False, id=16777395, type='Relic'), |  | ||||||
|     "King's_Idol-Glade_of_Hope": HKItemData(advancement=False, id=16777394, type='Relic'), |  | ||||||
|     "King's_Idol-Great_Hopper": HKItemData(advancement=False, id=16777396, type='Relic'), |  | ||||||
|     "King's_Idol-Grubs": HKItemData(advancement=False, id=16777391, type='Relic'), |  | ||||||
|     "King's_Idol-Pale_Lurker": HKItemData(advancement=False, id=16777397, type='Relic'), |  | ||||||
|     "King's_Pass": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "King's_Station_Stag": HKItemData(advancement=True, id=16777496, type='Stag'), |  | ||||||
|     'King_Fragment': HKItemData(advancement=True, id=16777277, type='Charm'), |  | ||||||
|     "Kingdom's_Edge_Map": HKItemData(advancement=False, id=16777483, type='Map'), |  | ||||||
|     'Lake_of_Unn': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Left_City': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Left_Elevator': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Left_Fog_Canyon': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Lifeblood_Cocoon-Ancestral_Mound': HKItemData(advancement=False, id=16777502, type='Cocoon'), |  | ||||||
|     'Lifeblood_Cocoon-Failed_Tramway': HKItemData(advancement=False, id=16777506, type='Cocoon'), |  | ||||||
|     'Lifeblood_Cocoon-Fog_Canyon_West': HKItemData(advancement=False, id=16777504, type='Cocoon'), |  | ||||||
|     'Lifeblood_Cocoon-Galien': HKItemData(advancement=False, id=16777507, type='Cocoon'), |  | ||||||
|     'Lifeblood_Cocoon-Greenpath': HKItemData(advancement=False, id=16777503, type='Cocoon'), |  | ||||||
|     "Lifeblood_Cocoon-King's_Pass": HKItemData(advancement=False, id=16777501, type='Cocoon'), |  | ||||||
|     "Lifeblood_Cocoon-Kingdom's_Edge": HKItemData(advancement=False, id=16777508, type='Cocoon'), |  | ||||||
|     'Lifeblood_Cocoon-Mantis_Village': HKItemData(advancement=False, id=16777505, type='Cocoon'), |  | ||||||
|     'Lifeblood_Core': HKItemData(advancement=True, id=16777249, type='Charm'), |  | ||||||
|     'Lifeblood_Heart': HKItemData(advancement=True, id=16777248, type='Charm'), |  | ||||||
|     'Longnail': HKItemData(advancement=False, id=16777258, type='Charm'), |  | ||||||
|     'Love_Key': HKItemData(advancement=True, id=16777292, type='Key'), |  | ||||||
|     'Lower_Basin': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Lower_King's_Station": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Lower_Kingdom's_Edge": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Lower_Left_Waterways': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Lower_Resting_Grounds': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Lower_Tram': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Lumafly_Lantern': HKItemData(advancement=True, id=16777284, type='Key'), |  | ||||||
|     'Lurien': HKItemData(advancement=True, id=16777217, type='Dreamer'), |  | ||||||
|     'Mantis_Claw': HKItemData(advancement=True, id=16777223, type='Skill'), |  | ||||||
|     'Mantis_Outskirts': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Mantis_Village': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Mark_of_Pride': HKItemData(advancement=True, id=16777253, type='Charm'), |  | ||||||
|     'Mask_Shard-5_Grubs': HKItemData(advancement=False, id=16777301, type='Mask'), |  | ||||||
|     'Mask_Shard-Bretta': HKItemData(advancement=False, id=16777311, type='Mask'), |  | ||||||
|     'Mask_Shard-Brooding_Mawlek': HKItemData(advancement=False, id=16777302, type='Mask'), |  | ||||||
|     'Mask_Shard-Crossroads_Goam': HKItemData(advancement=False, id=16777303, type='Mask'), |  | ||||||
|     'Mask_Shard-Deepnest': HKItemData(advancement=False, id=16777306, type='Mask'), |  | ||||||
|     'Mask_Shard-Enraged_Guardian': HKItemData(advancement=False, id=16777308, type='Mask'), |  | ||||||
|     'Mask_Shard-Grey_Mourner': HKItemData(advancement=False, id=16777310, type='Mask'), |  | ||||||
|     'Mask_Shard-Hive': HKItemData(advancement=False, id=16777309, type='Mask'), |  | ||||||
|     "Mask_Shard-Queen's_Station": HKItemData(advancement=False, id=16777305, type='Mask'), |  | ||||||
|     'Mask_Shard-Seer': HKItemData(advancement=False, id=16777300, type='Mask'), |  | ||||||
|     'Mask_Shard-Sly1': HKItemData(advancement=False, id=16777296, type='Mask'), |  | ||||||
|     'Mask_Shard-Sly2': HKItemData(advancement=False, id=16777297, type='Mask'), |  | ||||||
|     'Mask_Shard-Sly3': HKItemData(advancement=False, id=16777298, type='Mask'), |  | ||||||
|     'Mask_Shard-Sly4': HKItemData(advancement=False, id=16777299, type='Mask'), |  | ||||||
|     'Mask_Shard-Stone_Sanctuary': HKItemData(advancement=False, id=16777304, type='Mask'), |  | ||||||
|     'Mask_Shard-Waterways': HKItemData(advancement=False, id=16777307, type='Mask'), |  | ||||||
|     'Mid_Basin': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Monarch_Wings': HKItemData(advancement=True, id=16777225, type='Skill'), |  | ||||||
|     'Monomon': HKItemData(advancement=True, id=16777218, type='Dreamer'), |  | ||||||
|     'Mothwing_Cloak': HKItemData(advancement=True, id=16777222, type='Skill'), |  | ||||||
|     "Nailmaster's_Glory": HKItemData(advancement=False, id=16777266, type='Charm'), |  | ||||||
|     'Oro_Bench': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Overgrown_Mound': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Palace_Grounds': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Pale_Lurker_Area': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Pale_Ore-Basin': HKItemData(advancement=False, id=16777325, type='Ore'), |  | ||||||
|     'Pale_Ore-Colosseum': HKItemData(advancement=False, id=16777330, type='Ore'), |  | ||||||
|     'Pale_Ore-Crystal_Peak': HKItemData(advancement=False, id=16777326, type='Ore'), |  | ||||||
|     'Pale_Ore-Grubs': HKItemData(advancement=False, id=16777329, type='Ore'), |  | ||||||
|     'Pale_Ore-Nosk': HKItemData(advancement=False, id=16777327, type='Ore'), |  | ||||||
|     'Pale_Ore-Seer': HKItemData(advancement=False, id=16777328, type='Ore'), |  | ||||||
|     'Placeholder': HKItemData(advancement=False, id=16777522, type='Fake'), |  | ||||||
|     'Pleasure_House': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Queen's_Gardens_Map": HKItemData(advancement=False, id=16777488, type='Map'), |  | ||||||
|     "Queen's_Gardens_Stag": HKItemData(advancement=True, id=16777494, type='Stag'), |  | ||||||
|     "Queen's_Station": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Queen's_Station_Stag": HKItemData(advancement=True, id=16777493, type='Stag'), |  | ||||||
|     'Queen_Fragment': HKItemData(advancement=True, id=16777276, type='Charm'), |  | ||||||
|     'Quick_Focus': HKItemData(advancement=False, id=16777247, type='Charm'), |  | ||||||
|     'Quick_Slash': HKItemData(advancement=False, id=16777272, type='Charm'), |  | ||||||
|     "Rancid_Egg-Beast's_Den": HKItemData(advancement=False, id=16777351, type='Egg'), |  | ||||||
|     'Rancid_Egg-Blue_Lake': HKItemData(advancement=False, id=16777345, type='Egg'), |  | ||||||
|     'Rancid_Egg-City_of_Tears_Left': HKItemData(advancement=False, id=16777349, type='Egg'), |  | ||||||
|     'Rancid_Egg-City_of_Tears_Pleasure_House': HKItemData(advancement=False, id=16777350, type='Egg'), |  | ||||||
|     'Rancid_Egg-Crystal_Peak_Dark_Room': HKItemData(advancement=False, id=16777347, type='Egg'), |  | ||||||
|     'Rancid_Egg-Crystal_Peak_Dive_Entrance': HKItemData(advancement=False, id=16777346, type='Egg'), |  | ||||||
|     'Rancid_Egg-Crystal_Peak_Tall_Room': HKItemData(advancement=False, id=16777348, type='Egg'), |  | ||||||
|     'Rancid_Egg-Dark_Deepnest': HKItemData(advancement=False, id=16777352, type='Egg'), |  | ||||||
|     'Rancid_Egg-Fungal_Core': HKItemData(advancement=False, id=16777343, type='Egg'), |  | ||||||
|     'Rancid_Egg-Grubs': HKItemData(advancement=False, id=16777341, type='Egg'), |  | ||||||
|     'Rancid_Egg-Near_Quick_Slash': HKItemData(advancement=False, id=16777354, type='Egg'), |  | ||||||
|     "Rancid_Egg-Queen's_Gardens": HKItemData(advancement=False, id=16777344, type='Egg'), |  | ||||||
|     'Rancid_Egg-Sheo': HKItemData(advancement=False, id=16777342, type='Egg'), |  | ||||||
|     'Rancid_Egg-Sly': HKItemData(advancement=False, id=16777340, type='Egg'), |  | ||||||
|     "Rancid_Egg-Upper_Kingdom's_Edge": HKItemData(advancement=False, id=16777355, type='Egg'), |  | ||||||
|     'Rancid_Egg-Waterways_East': HKItemData(advancement=False, id=16777356, type='Egg'), |  | ||||||
|     'Rancid_Egg-Waterways_Main': HKItemData(advancement=False, id=16777357, type='Egg'), |  | ||||||
|     'Rancid_Egg-Waterways_West_Bluggsac': HKItemData(advancement=False, id=16777358, type='Egg'), |  | ||||||
|     'Rancid_Egg-Waterways_West_Pickup': HKItemData(advancement=False, id=16777359, type='Egg'), |  | ||||||
|     "Rancid_Egg-Weaver's_Den": HKItemData(advancement=False, id=16777353, type='Egg'), |  | ||||||
|     'Resting_Grounds_Map': HKItemData(advancement=False, id=16777489, type='Map'), |  | ||||||
|     'Resting_Grounds_Stag': HKItemData(advancement=True, id=16777497, type='Stag'), |  | ||||||
|     'Right_City': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Right_Elevator': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Right_Fog_Canyon': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Right_Waterways': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Royal_Waterways_Map': HKItemData(advancement=False, id=16777485, type='Map'), |  | ||||||
|     'Seer': HKItemData(advancement=False, id=16777520, type='Fake'), |  | ||||||
|     'Shade_Cloak': HKItemData(advancement=True, id=16777226, type='Skill'), |  | ||||||
|     'Shade_Soul': HKItemData(advancement=True, id=16777232, type='Skill'), |  | ||||||
|     'Shaman_Stone': HKItemData(advancement=False, id=16777259, type='Charm'), |  | ||||||
|     'Shape_of_Unn': HKItemData(advancement=False, id=16777268, type='Charm'), |  | ||||||
|     'Sharp_Shadow': HKItemData(advancement=True, id=16777256, type='Charm'), |  | ||||||
|     "Shopkeeper's_Key": HKItemData(advancement=True, id=16777290, type='Key'), |  | ||||||
|     'Simple_Key-Basin': HKItemData(advancement=True, id=16777287, type='Key'), |  | ||||||
|     'Simple_Key-City': HKItemData(advancement=True, id=16777288, type='Key'), |  | ||||||
|     'Simple_Key-Lurker': HKItemData(advancement=True, id=16777289, type='Key'), |  | ||||||
|     'Simple_Key-Sly': HKItemData(advancement=True, id=16777286, type='Key'), |  | ||||||
|     'Soul_Catcher': HKItemData(advancement=False, id=16777260, type='Charm'), |  | ||||||
|     'Soul_Eater': HKItemData(advancement=False, id=16777261, type='Charm'), |  | ||||||
|     'Soul_Sanctum': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Spell_Twister': HKItemData(advancement=False, id=16777273, type='Charm'), |  | ||||||
|     'Spirits_Glade': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Spore_Shroom': HKItemData(advancement=True, id=16777257, type='Charm'), |  | ||||||
|     'Sprintmaster': HKItemData(advancement=True, id=16777279, type='Charm'), |  | ||||||
|     'Stag_Nest': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Stag_Nest_Stag': HKItemData(advancement=True, id=16777500, type='Stag'), |  | ||||||
|     'Stalwart_Shell': HKItemData(advancement=False, id=16777244, type='Charm'), |  | ||||||
|     'Steady_Body': HKItemData(advancement=False, id=16777254, type='Charm'), |  | ||||||
|     'Stone_Sanctuary': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Teacher's_Archives": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Thorns_of_Agony': HKItemData(advancement=False, id=16777252, type='Charm'), |  | ||||||
|     "Top_Kingdom's_Edge": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Top_Left_Queen's_Gardens": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Top_Right_Queen's_Gardens": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Tower_of_Love': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Tram_Pass': HKItemData(advancement=True, id=16777285, type='Key'), |  | ||||||
|     'Upper_Basin': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Upper_Crystal_Peak': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Upper_Deepnest': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Upper_King's_Station": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     "Upper_Kingdom's_Edge": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Upper_Left_Waterways': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Upper_Resting_Grounds': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Upper_Tram': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Vengeful_Spirit': HKItemData(advancement=True, id=16777231, type='Skill'), |  | ||||||
|     'Vessel_Fragment-Basin': HKItemData(advancement=False, id=16777318, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-City': HKItemData(advancement=False, id=16777316, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Crossroads': HKItemData(advancement=False, id=16777317, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Deepnest': HKItemData(advancement=False, id=16777319, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Greenpath': HKItemData(advancement=False, id=16777315, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Seer': HKItemData(advancement=False, id=16777314, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Sly1': HKItemData(advancement=False, id=16777312, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Sly2': HKItemData(advancement=False, id=16777313, type='Vessel'), |  | ||||||
|     'Vessel_Fragment-Stag_Nest': HKItemData(advancement=False, id=16777320, type='Vessel'), |  | ||||||
|     'Void_Heart': HKItemData(advancement=True, id=16777278, type='Charm'), |  | ||||||
|     "Wanderer's_Journal-Above_Mantis_Village": HKItemData(advancement=False, id=16777364, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Ancient_Basin": HKItemData(advancement=False, id=16777370, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-City_Storerooms": HKItemData(advancement=False, id=16777369, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Cliffs": HKItemData(advancement=False, id=16777360, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Crystal_Peak_Crawlers": HKItemData(advancement=False, id=16777365, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Fungal_Wastes_Thorns_Gauntlet": HKItemData(advancement=False, id=16777363, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Greenpath_Lower": HKItemData(advancement=False, id=16777362, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Greenpath_Stag": HKItemData(advancement=False, id=16777361, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-King's_Station": HKItemData(advancement=False, id=16777367, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Kingdom's_Edge_Camp": HKItemData(advancement=False, id=16777372, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Kingdom's_Edge_Entrance": HKItemData(advancement=False, id=16777371, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Kingdom's_Edge_Requires_Dive": HKItemData(advancement=False, id=16777373, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Pleasure_House": HKItemData(advancement=False, id=16777368, type='Relic'), |  | ||||||
|     "Wanderer's_Journal-Resting_Grounds_Catacombs": HKItemData(advancement=False, id=16777366, type='Relic'), |  | ||||||
|     'Waterways_Shaft': HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Wayward_Compass': HKItemData(advancement=False, id=16777242, type='Charm'), |  | ||||||
|     "Weaver's_Den": HKItemData(advancement=True, id=0, type='Event'), |  | ||||||
|     'Weaversong': HKItemData(advancement=True, id=16777281, type='Charm'), |  | ||||||
|     'Whispering_Root-Ancestral_Mound': HKItemData(advancement=True, id=16777416, type='Root'), |  | ||||||
|     'Whispering_Root-City': HKItemData(advancement=True, id=16777411, type='Root'), |  | ||||||
|     'Whispering_Root-Crossroads': HKItemData(advancement=True, id=16777403, type='Root'), |  | ||||||
|     'Whispering_Root-Crystal_Peak': HKItemData(advancement=True, id=16777414, type='Root'), |  | ||||||
|     'Whispering_Root-Deepnest': HKItemData(advancement=True, id=16777407, type='Root'), |  | ||||||
|     'Whispering_Root-Greenpath': HKItemData(advancement=True, id=16777404, type='Root'), |  | ||||||
|     'Whispering_Root-Hive': HKItemData(advancement=True, id=16777417, type='Root'), |  | ||||||
|     'Whispering_Root-Howling_Cliffs': HKItemData(advancement=True, id=16777415, type='Root'), |  | ||||||
|     'Whispering_Root-Kingdoms_Edge': HKItemData(advancement=True, id=16777409, type='Root'), |  | ||||||
|     'Whispering_Root-Leg_Eater': HKItemData(advancement=True, id=16777405, type='Root'), |  | ||||||
|     'Whispering_Root-Mantis_Village': HKItemData(advancement=True, id=16777406, type='Root'), |  | ||||||
|     'Whispering_Root-Queens_Gardens': HKItemData(advancement=True, id=16777408, type='Root'), |  | ||||||
|     'Whispering_Root-Resting_Grounds': HKItemData(advancement=True, id=16777412, type='Root'), |  | ||||||
|     'Whispering_Root-Spirits_Glade': HKItemData(advancement=True, id=16777413, type='Root'), |  | ||||||
|     'Whispering_Root-Waterways': HKItemData(advancement=True, id=16777410, type='Root'), |  | ||||||
|     'World_Sense': HKItemData(advancement=False, id=16777220, type='Dreamer')} |  | ||||||
|  |  | ||||||
| lookup_id_to_name:Dict[int, str] = {data.id: item_name for item_name, data in item_table.items() if data.type != 'Event'} | class HKItemData(NamedTuple): | ||||||
|  |     advancement: bool | ||||||
|  |     id: int | ||||||
|  |     type: str | ||||||
|  |  | ||||||
|  |  | ||||||
|  | for i, (item_name, item_type) in enumerate(items.items(), start=0x1000000): | ||||||
|  |     item_table[item_name] = HKItemData(advancement=item_name in logic_items or item_name in item_effects, | ||||||
|  |                                        id=i, type=item_type) | ||||||
|  |  | ||||||
|  | lookup_id_to_name: Dict[int, str] = {data.id: item_name for item_name, data in item_table.items()} | ||||||
| lookup_type_to_names: Dict[str, Set[str]] = {} | lookup_type_to_names: Dict[str, Set[str]] = {} | ||||||
| for item, item_data in item_table.items(): | for item, item_data in item_table.items(): | ||||||
|     lookup_type_to_names.setdefault(item_data.type, set()).add(item) |     lookup_type_to_names.setdefault(item_data.type, set()).add(item) | ||||||
| @@ -1,308 +0,0 @@ | |||||||
| # generated by https://github.com/Berserker66/HollowKnight.RandomizerMod/blob/master/extract_data.py |  | ||||||
| # do not edit manually |  | ||||||
|  |  | ||||||
| lookup_id_to_name = \ |  | ||||||
| {   17825793: 'Lurien', |  | ||||||
|     17825794: 'Monomon', |  | ||||||
|     17825795: 'Herrah', |  | ||||||
|     17825796: 'World_Sense', |  | ||||||
|     17825798: 'Mothwing_Cloak', |  | ||||||
|     17825799: 'Mantis_Claw', |  | ||||||
|     17825800: 'Crystal_Heart', |  | ||||||
|     17825801: 'Monarch_Wings', |  | ||||||
|     17825802: 'Shade_Cloak', |  | ||||||
|     17825803: "Isma's_Tear", |  | ||||||
|     17825804: 'Dream_Nail', |  | ||||||
|     17825805: 'Dream_Gate', |  | ||||||
|     17825806: 'Awoken_Dream_Nail', |  | ||||||
|     17825807: 'Vengeful_Spirit', |  | ||||||
|     17825808: 'Shade_Soul', |  | ||||||
|     17825809: 'Desolate_Dive', |  | ||||||
|     17825810: 'Descending_Dark', |  | ||||||
|     17825811: 'Howling_Wraiths', |  | ||||||
|     17825812: 'Abyss_Shriek', |  | ||||||
|     17825813: 'Cyclone_Slash', |  | ||||||
|     17825814: 'Dash_Slash', |  | ||||||
|     17825815: 'Great_Slash', |  | ||||||
|     17825816: 'Focus', |  | ||||||
|     17825817: 'Gathering_Swarm', |  | ||||||
|     17825818: 'Wayward_Compass', |  | ||||||
|     17825819: 'Grubsong', |  | ||||||
|     17825820: 'Stalwart_Shell', |  | ||||||
|     17825821: 'Baldur_Shell', |  | ||||||
|     17825822: 'Fury_of_the_Fallen', |  | ||||||
|     17825823: 'Quick_Focus', |  | ||||||
|     17825824: 'Lifeblood_Heart', |  | ||||||
|     17825825: 'Lifeblood_Core', |  | ||||||
|     17825826: "Defender's_Crest", |  | ||||||
|     17825827: 'Flukenest', |  | ||||||
|     17825828: 'Thorns_of_Agony', |  | ||||||
|     17825829: 'Mark_of_Pride', |  | ||||||
|     17825830: 'Steady_Body', |  | ||||||
|     17825831: 'Heavy_Blow', |  | ||||||
|     17825832: 'Sharp_Shadow', |  | ||||||
|     17825833: 'Spore_Shroom', |  | ||||||
|     17825834: 'Longnail', |  | ||||||
|     17825835: 'Shaman_Stone', |  | ||||||
|     17825836: 'Soul_Catcher', |  | ||||||
|     17825837: 'Soul_Eater', |  | ||||||
|     17825838: 'Glowing_Womb', |  | ||||||
|     17825839: 'Fragile_Heart', |  | ||||||
|     17825840: 'Fragile_Greed', |  | ||||||
|     17825841: 'Fragile_Strength', |  | ||||||
|     17825842: "Nailmaster's_Glory", |  | ||||||
|     17825843: "Joni's_Blessing", |  | ||||||
|     17825844: 'Shape_of_Unn', |  | ||||||
|     17825845: 'Hiveblood', |  | ||||||
|     17825846: 'Dream_Wielder', |  | ||||||
|     17825847: 'Dashmaster', |  | ||||||
|     17825848: 'Quick_Slash', |  | ||||||
|     17825849: 'Spell_Twister', |  | ||||||
|     17825850: 'Deep_Focus', |  | ||||||
|     17825851: "Grubberfly's_Elegy", |  | ||||||
|     17825852: 'Queen_Fragment', |  | ||||||
|     17825853: 'King_Fragment', |  | ||||||
|     17825854: 'Void_Heart', |  | ||||||
|     17825855: 'Sprintmaster', |  | ||||||
|     17825856: 'Dreamshield', |  | ||||||
|     17825857: 'Weaversong', |  | ||||||
|     17825858: 'Grimmchild', |  | ||||||
|     17825859: 'City_Crest', |  | ||||||
|     17825860: 'Lumafly_Lantern', |  | ||||||
|     17825861: 'Tram_Pass', |  | ||||||
|     17825862: 'Simple_Key-Sly', |  | ||||||
|     17825863: 'Simple_Key-Basin', |  | ||||||
|     17825864: 'Simple_Key-City', |  | ||||||
|     17825865: 'Simple_Key-Lurker', |  | ||||||
|     17825866: "Shopkeeper's_Key", |  | ||||||
|     17825867: 'Elegant_Key', |  | ||||||
|     17825868: 'Love_Key', |  | ||||||
|     17825869: "King's_Brand", |  | ||||||
|     17825870: 'Godtuner', |  | ||||||
|     17825871: "Collector's_Map", |  | ||||||
|     17825872: 'Mask_Shard-Sly1', |  | ||||||
|     17825873: 'Mask_Shard-Sly2', |  | ||||||
|     17825874: 'Mask_Shard-Sly3', |  | ||||||
|     17825875: 'Mask_Shard-Sly4', |  | ||||||
|     17825876: 'Mask_Shard-Seer', |  | ||||||
|     17825877: 'Mask_Shard-5_Grubs', |  | ||||||
|     17825878: 'Mask_Shard-Brooding_Mawlek', |  | ||||||
|     17825879: 'Mask_Shard-Crossroads_Goam', |  | ||||||
|     17825880: 'Mask_Shard-Stone_Sanctuary', |  | ||||||
|     17825881: "Mask_Shard-Queen's_Station", |  | ||||||
|     17825882: 'Mask_Shard-Deepnest', |  | ||||||
|     17825883: 'Mask_Shard-Waterways', |  | ||||||
|     17825884: 'Mask_Shard-Enraged_Guardian', |  | ||||||
|     17825885: 'Mask_Shard-Hive', |  | ||||||
|     17825886: 'Mask_Shard-Grey_Mourner', |  | ||||||
|     17825887: 'Mask_Shard-Bretta', |  | ||||||
|     17825888: 'Vessel_Fragment-Sly1', |  | ||||||
|     17825889: 'Vessel_Fragment-Sly2', |  | ||||||
|     17825890: 'Vessel_Fragment-Seer', |  | ||||||
|     17825891: 'Vessel_Fragment-Greenpath', |  | ||||||
|     17825892: 'Vessel_Fragment-City', |  | ||||||
|     17825893: 'Vessel_Fragment-Crossroads', |  | ||||||
|     17825894: 'Vessel_Fragment-Basin', |  | ||||||
|     17825895: 'Vessel_Fragment-Deepnest', |  | ||||||
|     17825896: 'Vessel_Fragment-Stag_Nest', |  | ||||||
|     17825897: 'Charm_Notch-Shrumal_Ogres', |  | ||||||
|     17825898: 'Charm_Notch-Fog_Canyon', |  | ||||||
|     17825899: 'Charm_Notch-Colosseum', |  | ||||||
|     17825900: 'Charm_Notch-Grimm', |  | ||||||
|     17825901: 'Pale_Ore-Basin', |  | ||||||
|     17825902: 'Pale_Ore-Crystal_Peak', |  | ||||||
|     17825903: 'Pale_Ore-Nosk', |  | ||||||
|     17825904: 'Pale_Ore-Seer', |  | ||||||
|     17825905: 'Pale_Ore-Grubs', |  | ||||||
|     17825906: 'Pale_Ore-Colosseum', |  | ||||||
|     17825907: '200_Geo-False_Knight_Chest', |  | ||||||
|     17825908: '380_Geo-Soul_Master_Chest', |  | ||||||
|     17825909: '655_Geo-Watcher_Knights_Chest', |  | ||||||
|     17825910: '85_Geo-Greenpath_Chest', |  | ||||||
|     17825911: '620_Geo-Mantis_Lords_Chest', |  | ||||||
|     17825912: '150_Geo-Resting_Grounds_Chest', |  | ||||||
|     17825913: '80_Geo-Crystal_Peak_Chest', |  | ||||||
|     17825914: '160_Geo-Weavers_Den_Chest', |  | ||||||
|     17825916: 'Rancid_Egg-Sly', |  | ||||||
|     17825917: 'Rancid_Egg-Grubs', |  | ||||||
|     17825918: 'Rancid_Egg-Sheo', |  | ||||||
|     17825919: 'Rancid_Egg-Fungal_Core', |  | ||||||
|     17825920: "Rancid_Egg-Queen's_Gardens", |  | ||||||
|     17825921: 'Rancid_Egg-Blue_Lake', |  | ||||||
|     17825922: 'Rancid_Egg-Crystal_Peak_Dive_Entrance', |  | ||||||
|     17825923: 'Rancid_Egg-Crystal_Peak_Dark_Room', |  | ||||||
|     17825924: 'Rancid_Egg-Crystal_Peak_Tall_Room', |  | ||||||
|     17825925: 'Rancid_Egg-City_of_Tears_Left', |  | ||||||
|     17825926: 'Rancid_Egg-City_of_Tears_Pleasure_House', |  | ||||||
|     17825927: "Rancid_Egg-Beast's_Den", |  | ||||||
|     17825928: 'Rancid_Egg-Dark_Deepnest', |  | ||||||
|     17825929: "Rancid_Egg-Weaver's_Den", |  | ||||||
|     17825930: 'Rancid_Egg-Near_Quick_Slash', |  | ||||||
|     17825931: "Rancid_Egg-Upper_Kingdom's_Edge", |  | ||||||
|     17825932: 'Rancid_Egg-Waterways_East', |  | ||||||
|     17825933: 'Rancid_Egg-Waterways_Main', |  | ||||||
|     17825934: 'Rancid_Egg-Waterways_West_Bluggsac', |  | ||||||
|     17825935: 'Rancid_Egg-Waterways_West_Pickup', |  | ||||||
|     17825936: "Wanderer's_Journal-Cliffs", |  | ||||||
|     17825937: "Wanderer's_Journal-Greenpath_Stag", |  | ||||||
|     17825938: "Wanderer's_Journal-Greenpath_Lower", |  | ||||||
|     17825939: "Wanderer's_Journal-Fungal_Wastes_Thorns_Gauntlet", |  | ||||||
|     17825940: "Wanderer's_Journal-Above_Mantis_Village", |  | ||||||
|     17825941: "Wanderer's_Journal-Crystal_Peak_Crawlers", |  | ||||||
|     17825942: "Wanderer's_Journal-Resting_Grounds_Catacombs", |  | ||||||
|     17825943: "Wanderer's_Journal-King's_Station", |  | ||||||
|     17825944: "Wanderer's_Journal-Pleasure_House", |  | ||||||
|     17825945: "Wanderer's_Journal-City_Storerooms", |  | ||||||
|     17825946: "Wanderer's_Journal-Ancient_Basin", |  | ||||||
|     17825947: "Wanderer's_Journal-Kingdom's_Edge_Entrance", |  | ||||||
|     17825948: "Wanderer's_Journal-Kingdom's_Edge_Camp", |  | ||||||
|     17825949: "Wanderer's_Journal-Kingdom's_Edge_Requires_Dive", |  | ||||||
|     17825950: 'Hallownest_Seal-Crossroads_Well', |  | ||||||
|     17825951: 'Hallownest_Seal-Grubs', |  | ||||||
|     17825952: 'Hallownest_Seal-Greenpath', |  | ||||||
|     17825953: 'Hallownest_Seal-Fog_Canyon_West', |  | ||||||
|     17825954: 'Hallownest_Seal-Fog_Canyon_East', |  | ||||||
|     17825955: "Hallownest_Seal-Queen's_Station", |  | ||||||
|     17825956: 'Hallownest_Seal-Fungal_Wastes_Sporgs', |  | ||||||
|     17825957: 'Hallownest_Seal-Mantis_Lords', |  | ||||||
|     17825958: 'Hallownest_Seal-Seer', |  | ||||||
|     17825959: 'Hallownest_Seal-Resting_Grounds_Catacombs', |  | ||||||
|     17825960: "Hallownest_Seal-King's_Station", |  | ||||||
|     17825961: 'Hallownest_Seal-City_Rafters', |  | ||||||
|     17825962: 'Hallownest_Seal-Soul_Sanctum', |  | ||||||
|     17825963: 'Hallownest_Seal-Watcher_Knight', |  | ||||||
|     17825964: 'Hallownest_Seal-Deepnest_By_Mantis_Lords', |  | ||||||
|     17825965: "Hallownest_Seal-Beast's_Den", |  | ||||||
|     17825966: "Hallownest_Seal-Queen's_Gardens", |  | ||||||
|     17825967: "King's_Idol-Grubs", |  | ||||||
|     17825968: "King's_Idol-Cliffs", |  | ||||||
|     17825969: "King's_Idol-Crystal_Peak", |  | ||||||
|     17825970: "King's_Idol-Glade_of_Hope", |  | ||||||
|     17825971: "King's_Idol-Dung_Defender", |  | ||||||
|     17825972: "King's_Idol-Great_Hopper", |  | ||||||
|     17825973: "King's_Idol-Pale_Lurker", |  | ||||||
|     17825974: "King's_Idol-Deepnest", |  | ||||||
|     17825975: 'Arcane_Egg-Seer', |  | ||||||
|     17825976: 'Arcane_Egg-Lifeblood_Core', |  | ||||||
|     17825977: 'Arcane_Egg-Shade_Cloak', |  | ||||||
|     17825978: 'Arcane_Egg-Birthplace', |  | ||||||
|     17825979: 'Whispering_Root-Crossroads', |  | ||||||
|     17825980: 'Whispering_Root-Greenpath', |  | ||||||
|     17825981: 'Whispering_Root-Leg_Eater', |  | ||||||
|     17825982: 'Whispering_Root-Mantis_Village', |  | ||||||
|     17825983: 'Whispering_Root-Deepnest', |  | ||||||
|     17825984: 'Whispering_Root-Queens_Gardens', |  | ||||||
|     17825985: 'Whispering_Root-Kingdoms_Edge', |  | ||||||
|     17825986: 'Whispering_Root-Waterways', |  | ||||||
|     17825987: 'Whispering_Root-City', |  | ||||||
|     17825988: 'Whispering_Root-Resting_Grounds', |  | ||||||
|     17825989: 'Whispering_Root-Spirits_Glade', |  | ||||||
|     17825990: 'Whispering_Root-Crystal_Peak', |  | ||||||
|     17825991: 'Whispering_Root-Howling_Cliffs', |  | ||||||
|     17825992: 'Whispering_Root-Ancestral_Mound', |  | ||||||
|     17825993: 'Whispering_Root-Hive', |  | ||||||
|     17825994: 'Boss_Essence-Elder_Hu', |  | ||||||
|     17825995: 'Boss_Essence-Xero', |  | ||||||
|     17825996: 'Boss_Essence-Gorb', |  | ||||||
|     17825997: 'Boss_Essence-Marmu', |  | ||||||
|     17825998: 'Boss_Essence-No_Eyes', |  | ||||||
|     17825999: 'Boss_Essence-Galien', |  | ||||||
|     17826000: 'Boss_Essence-Markoth', |  | ||||||
|     17826001: 'Boss_Essence-Failed_Champion', |  | ||||||
|     17826002: 'Boss_Essence-Soul_Tyrant', |  | ||||||
|     17826003: 'Boss_Essence-Lost_Kin', |  | ||||||
|     17826004: 'Boss_Essence-White_Defender', |  | ||||||
|     17826005: 'Boss_Essence-Grey_Prince_Zote', |  | ||||||
|     17826006: 'Grub-Crossroads_Acid', |  | ||||||
|     17826007: 'Grub-Crossroads_Center', |  | ||||||
|     17826008: 'Grub-Crossroads_Stag', |  | ||||||
|     17826009: 'Grub-Crossroads_Spike', |  | ||||||
|     17826010: 'Grub-Crossroads_Guarded', |  | ||||||
|     17826011: 'Grub-Greenpath_Cornifer', |  | ||||||
|     17826012: 'Grub-Greenpath_Journal', |  | ||||||
|     17826013: 'Grub-Greenpath_MMC', |  | ||||||
|     17826014: 'Grub-Greenpath_Stag', |  | ||||||
|     17826015: 'Grub-Fog_Canyon', |  | ||||||
|     17826016: 'Grub-Fungal_Bouncy', |  | ||||||
|     17826017: 'Grub-Fungal_Spore_Shroom', |  | ||||||
|     17826018: 'Grub-Deepnest_Mimic', |  | ||||||
|     17826019: 'Grub-Deepnest_Nosk', |  | ||||||
|     17826020: 'Grub-Deepnest_Spike', |  | ||||||
|     17826021: 'Grub-Dark_Deepnest', |  | ||||||
|     17826022: "Grub-Beast's_Den", |  | ||||||
|     17826023: "Grub-Kingdom's_Edge_Oro", |  | ||||||
|     17826024: "Grub-Kingdom's_Edge_Camp", |  | ||||||
|     17826025: 'Grub-Hive_External', |  | ||||||
|     17826026: 'Grub-Hive_Internal', |  | ||||||
|     17826027: 'Grub-Basin_Requires_Wings', |  | ||||||
|     17826028: 'Grub-Basin_Requires_Dive', |  | ||||||
|     17826029: 'Grub-Waterways_Main', |  | ||||||
|     17826030: 'Grub-Waterways_East', |  | ||||||
|     17826031: 'Grub-Waterways_Requires_Tram', |  | ||||||
|     17826032: 'Grub-City_of_Tears_Left', |  | ||||||
|     17826033: 'Grub-Soul_Sanctum', |  | ||||||
|     17826034: "Grub-Watcher's_Spire", |  | ||||||
|     17826035: 'Grub-City_of_Tears_Guarded', |  | ||||||
|     17826036: "Grub-King's_Station", |  | ||||||
|     17826037: 'Grub-Resting_Grounds', |  | ||||||
|     17826038: 'Grub-Crystal_Peak_Below_Chest', |  | ||||||
|     17826039: 'Grub-Crystallized_Mound', |  | ||||||
|     17826040: 'Grub-Crystal_Peak_Spike', |  | ||||||
|     17826041: 'Grub-Crystal_Peak_Mimic', |  | ||||||
|     17826042: 'Grub-Crystal_Peak_Crushers', |  | ||||||
|     17826043: 'Grub-Crystal_Heart', |  | ||||||
|     17826044: 'Grub-Hallownest_Crown', |  | ||||||
|     17826045: 'Grub-Howling_Cliffs', |  | ||||||
|     17826046: "Grub-Queen's_Gardens_Stag", |  | ||||||
|     17826047: "Grub-Queen's_Gardens_Marmu", |  | ||||||
|     17826048: "Grub-Queen's_Gardens_Top", |  | ||||||
|     17826049: 'Grub-Collector_1', |  | ||||||
|     17826050: 'Grub-Collector_2', |  | ||||||
|     17826051: 'Grub-Collector_3', |  | ||||||
|     17826052: 'Crossroads_Map', |  | ||||||
|     17826053: 'Greenpath_Map', |  | ||||||
|     17826054: 'Fog_Canyon_Map', |  | ||||||
|     17826055: 'Fungal_Wastes_Map', |  | ||||||
|     17826056: 'Deepnest_Map-Upper', |  | ||||||
|     17826057: 'Deepnest_Map-Right_[Gives_Quill]', |  | ||||||
|     17826058: 'Ancient_Basin_Map', |  | ||||||
|     17826059: "Kingdom's_Edge_Map", |  | ||||||
|     17826060: 'City_of_Tears_Map', |  | ||||||
|     17826061: 'Royal_Waterways_Map', |  | ||||||
|     17826062: 'Howling_Cliffs_Map', |  | ||||||
|     17826063: 'Crystal_Peak_Map', |  | ||||||
|     17826064: "Queen's_Gardens_Map", |  | ||||||
|     17826065: 'Resting_Grounds_Map', |  | ||||||
|     17826066: 'Dirtmouth_Stag', |  | ||||||
|     17826067: 'Crossroads_Stag', |  | ||||||
|     17826068: 'Greenpath_Stag', |  | ||||||
|     17826069: "Queen's_Station_Stag", |  | ||||||
|     17826070: "Queen's_Gardens_Stag", |  | ||||||
|     17826071: 'City_Storerooms_Stag', |  | ||||||
|     17826072: "King's_Station_Stag", |  | ||||||
|     17826073: 'Resting_Grounds_Stag', |  | ||||||
|     17826074: 'Distant_Village_Stag', |  | ||||||
|     17826075: 'Hidden_Station_Stag', |  | ||||||
|     17826076: 'Stag_Nest_Stag', |  | ||||||
|     17826077: "Lifeblood_Cocoon-King's_Pass", |  | ||||||
|     17826078: 'Lifeblood_Cocoon-Ancestral_Mound', |  | ||||||
|     17826079: 'Lifeblood_Cocoon-Greenpath', |  | ||||||
|     17826080: 'Lifeblood_Cocoon-Fog_Canyon_West', |  | ||||||
|     17826081: 'Lifeblood_Cocoon-Mantis_Village', |  | ||||||
|     17826082: 'Lifeblood_Cocoon-Failed_Tramway', |  | ||||||
|     17826083: 'Lifeblood_Cocoon-Galien', |  | ||||||
|     17826084: "Lifeblood_Cocoon-Kingdom's_Edge", |  | ||||||
|     17826085: 'Grimmkin_Flame-City_Storerooms', |  | ||||||
|     17826086: 'Grimmkin_Flame-Greenpath', |  | ||||||
|     17826087: 'Grimmkin_Flame-Crystal_Peak', |  | ||||||
|     17826088: "Grimmkin_Flame-King's_Pass", |  | ||||||
|     17826089: 'Grimmkin_Flame-Resting_Grounds', |  | ||||||
|     17826090: "Grimmkin_Flame-Kingdom's_Edge", |  | ||||||
|     17826091: 'Grimmkin_Flame-Fungal_Core', |  | ||||||
|     17826092: 'Grimmkin_Flame-Ancient_Basin', |  | ||||||
|     17826093: 'Grimmkin_Flame-Hive', |  | ||||||
|     17826094: 'Grimmkin_Flame-Brumm'} |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| lookup_name_to_id = {location_name: location_id for location_id, location_name in lookup_id_to_name.items()} |  | ||||||
| @@ -1,39 +1,138 @@ | |||||||
| import typing | import typing | ||||||
|  | from .ExtractedData import logic_options, starts, pool_options | ||||||
|  | from Options import Option, DefaultOnToggle, Toggle, Choice, Range | ||||||
|  |  | ||||||
| from Options import Option, DefaultOnToggle, Toggle | # This way the dynamic start names are picked up by the MetaClass Choice belongs to | ||||||
|  | StartLocation = type("StartLocation", (Choice,), { | ||||||
|  |     "option_" + start: i for i, start in enumerate(starts)} | {"auto_display_name": False}) | ||||||
|  |  | ||||||
| hollow_knight_randomize_options: typing.Dict[str, type(Option)] = { | default_on = { | ||||||
|     "RandomizeDreamers": DefaultOnToggle, |     "RandomizeDreamers", | ||||||
|     "RandomizeSkills": DefaultOnToggle, |     "RandomizeSkills", | ||||||
|     "RandomizeCharms": DefaultOnToggle, |     "RandomizeCharms", | ||||||
|     "RandomizeKeys": DefaultOnToggle, |     "RandomizeKeys", | ||||||
|     "RandomizeGeoChests": Toggle, |     "RandomizeMaskShards", | ||||||
|     "RandomizeMaskShards": DefaultOnToggle, |     "RandomizeVesselFragments", | ||||||
|     "RandomizeVesselFragments": DefaultOnToggle, |     "RandomizePaleOre", | ||||||
|     "RandomizeCharmNotches": Toggle, |     "RandomizeRelics" | ||||||
|     "RandomizePaleOre": DefaultOnToggle, |  | ||||||
|     "RandomizeRancidEggs": Toggle, |  | ||||||
|     "RandomizeRelics": DefaultOnToggle, |  | ||||||
|     "RandomizeMaps": Toggle, |  | ||||||
|     "RandomizeStags": Toggle, |  | ||||||
|     "RandomizeGrubs": Toggle, |  | ||||||
|     "RandomizeWhisperingRoots": Toggle, |  | ||||||
|     "RandomizeRocks": Toggle, |  | ||||||
|     "RandomizeSoulTotems": Toggle, |  | ||||||
|     "RandomizePalaceTotems": Toggle, |  | ||||||
|     "RandomizeLoreTablets": Toggle, |  | ||||||
|     "RandomizeLifebloodCocoons": Toggle, |  | ||||||
|     "RandomizeFlames": Toggle |  | ||||||
| } | } | ||||||
| hollow_knight_skip_options: typing.Dict[str, type(Option)] = { |  | ||||||
|     "MILDSKIPS": Toggle, | # not supported at this time | ||||||
|     "SPICYSKIPS": Toggle, | disabled = { | ||||||
|     "FIREBALLSKIPS": Toggle, |     "RandomizeFocus", | ||||||
|     "ACIDSKIPS": Toggle, |     "RandomizeSwim", | ||||||
|     "SPIKETUNNELS": Toggle, |     "RandomizeMimics", | ||||||
|     "DARKROOMS": Toggle, |     "RandomizeNail", | ||||||
|     "CURSED": Toggle, |  | ||||||
|     "SHADESKIPS": Toggle, | } | ||||||
|  |  | ||||||
|  | hollow_knight_randomize_options: typing.Dict[str, type(Option)] = {} | ||||||
|  |  | ||||||
|  | for option_name, option_data in pool_options.items(): | ||||||
|  |     extra_data = {"items": option_data[0], "locations": option_data[1]} | ||||||
|  |     if option_name in default_on: | ||||||
|  |         option = type(option_name, (DefaultOnToggle,), extra_data) | ||||||
|  |     else: | ||||||
|  |         option = type(option_name, (Toggle,), extra_data) | ||||||
|  |     hollow_knight_randomize_options[option_name] = option | ||||||
|  |  | ||||||
|  | hollow_knight_logic_options: typing.Dict[str, type(Option)] = { | ||||||
|  |     option_name: Toggle for option_name in logic_options.values() if | ||||||
|  |     option_name not in hollow_knight_randomize_options | ||||||
|  |     and option_name != "RandomizeCharmNotches"} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MinimumGrubPrice(Range): | ||||||
|  |     display_name = "Minimum Grub Price" | ||||||
|  |     range_start = 1 | ||||||
|  |     range_end = 46 | ||||||
|  |     default = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MaximumGrubPrice(MinimumGrubPrice): | ||||||
|  |     display_name = "Maximum Grub Price" | ||||||
|  |     default = 23 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MinimumEssencePrice(Range): | ||||||
|  |     display_name = "Minimum Essence Price" | ||||||
|  |     range_start = 1 | ||||||
|  |     range_end = 2800 | ||||||
|  |     default = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MaximumEssencePrice(MinimumEssencePrice): | ||||||
|  |     display_name = "Maximum Essence Price" | ||||||
|  |     default = 1400 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MinimumEggPrice(Range): | ||||||
|  |     display_name = "Minimum Egg Price" | ||||||
|  |     range_start = 1 | ||||||
|  |     range_end = 21 | ||||||
|  |     default = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MaximumEggPrice(MinimumEggPrice): | ||||||
|  |     display_name = "Maximum Egg Price" | ||||||
|  |     default = 10 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MinimumCharmPrice(Range): | ||||||
|  |     """For Salubra's Charm-count based locations.""" | ||||||
|  |     display_name = "Minimum Charm Requirement" | ||||||
|  |     range_start = 1 | ||||||
|  |     range_end = 40 | ||||||
|  |     default = 1 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MaximumCharmPrice(MinimumCharmPrice): | ||||||
|  |     default = 20 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class RandomCharmCosts(Range): | ||||||
|  |     """Total Cost of all Charms together. Set to -1 for vanilla costs. Vanilla sums to 90.""" | ||||||
|  |  | ||||||
|  |     display_name = "Random Charm Costs" | ||||||
|  |     range_start = -1 | ||||||
|  |     range_end = 240 | ||||||
|  |     default = -1 | ||||||
|  |     vanilla_costs: typing.List[int] = [1, 1, 1, 2, 2, 2, 3, 2, 3, 1, 3, 1, 3, 1, 2, 2, 1, 2, 3, 2, | ||||||
|  |                                        4, 2, 2, 2, 3, 1, 4, 2, 4, 1, 2, 3, 2, 4, 3, 5, 1, 3, 2, 2] | ||||||
|  |     charm_count: int = len(vanilla_costs) | ||||||
|  |  | ||||||
|  |     def get_costs(self, random_source) -> typing.List[int]: | ||||||
|  |         if -1 == self.value: | ||||||
|  |             return self.vanilla_costs | ||||||
|  |         else: | ||||||
|  |             charms = [0]*self.charm_count | ||||||
|  |             for x in range(self.value): | ||||||
|  |                 index = random_source.randint(0, self.charm_count-1) | ||||||
|  |                 while charms[index] > 5: | ||||||
|  |                     index = random_source.randint(0, self.charm_count-1) | ||||||
|  |                 charms[index] += 1 | ||||||
|  |             return charms | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class EggShopSlots(Range): | ||||||
|  |     """For each slot, add a location to the Egg Shop and a Geo drop to the item pool.""" | ||||||
|  |  | ||||||
|  |     display_name = "Egg Shop Item Slots" | ||||||
|  |     range_end = 16 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | hollow_knight_options: typing.Dict[str, type(Option)] = { | ||||||
|  |     **hollow_knight_randomize_options, | ||||||
|  |     **hollow_knight_logic_options, | ||||||
|  |     "start_location": StartLocation, | ||||||
|  |     "minimum_grub_price": MinimumGrubPrice, | ||||||
|  |     "maximum_grub_price": MaximumGrubPrice, | ||||||
|  |     "minimum_essence_price": MinimumEssencePrice, | ||||||
|  |     "maximum_essence_price": MaximumEssencePrice, | ||||||
|  |     "minimum_egg_price": MinimumEggPrice, | ||||||
|  |     "maximum_egg_price": MaximumEggPrice, | ||||||
|  |     "minimum_charm_price": MinimumCharmPrice, | ||||||
|  |     "maximum_charm_price": MaximumCharmPrice, | ||||||
|  |     "random_charm_costs": RandomCharmCosts, | ||||||
|  |     "egg_shop_slots": EggShopSlots, | ||||||
| } | } | ||||||
| hollow_knight_options: typing.Dict[str, type(Option)] = {**hollow_knight_randomize_options, |  | ||||||
|                                                          **hollow_knight_skip_options} |  | ||||||
| @@ -1,13 +1,25 @@ | |||||||
| # generated by https://github.com/Berserker66/HollowKnight.RandomizerMod/blob/master/extract_data.py | from .ExtractedData import region_names, exits, connectors | ||||||
| # do not edit manually |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def create_regions(world, player: int): | def create_regions(world, player: int): | ||||||
|     from . import create_region |     from . import create_region, HKLocation, HKItem | ||||||
|     from .Items import item_table |     world.regions.append(create_region(world, player, 'Menu', None, ['Hollow Nest S&Q'])) | ||||||
|     from .Locations import lookup_name_to_id |     for region in region_names: | ||||||
|     world.regions += [ |         world.regions.append(create_region(world, player, region, [], | ||||||
|         create_region(world, player, 'Menu', None, ['Hollow Nest S&Q']), |                                            exits.get(region, []))) | ||||||
|         create_region(world, player, 'Hollow Nest', [location for location in lookup_name_to_id] + |     for entrance_name, exit_name in connectors.items(): | ||||||
|                       [item_name for item_name, item_data in item_table.items() if item_data.type == "Event"]) |         if exit_name: | ||||||
|     ] |             target_region = world.get_entrance(exit_name, player).parent_region | ||||||
|  |             world.get_entrance(entrance_name, player).connect(target_region) | ||||||
|  |             if not entrance_name.endswith("_R"): | ||||||
|  |                 # a traversable entrance puts the name of the target door "into logic". | ||||||
|  |                 loc = HKLocation(player, exit_name, None, target_region) | ||||||
|  |                 loc.place_locked_item(HKItem(exit_name, | ||||||
|  |                                              not exit_name.startswith("White_Palace_"), | ||||||
|  |                                              None, "Event", player)) | ||||||
|  |                 target_region.locations.append(loc) | ||||||
|  |         else: | ||||||
|  |             ent = world.get_entrance(entrance_name, player) | ||||||
|  |             ent.parent_region.exits.remove(ent) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2088
									
								
								worlds/hk/Rules.py
									
									
									
									
									
								
							
							
						
						
									
										2088
									
								
								worlds/hk/Rules.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,7 +0,0 @@ | |||||||
| import typing |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class HKItemData(typing.NamedTuple): |  | ||||||
|     advancement: bool |  | ||||||
|     id: int |  | ||||||
|     type: str |  | ||||||
| @@ -1,111 +1,262 @@ | |||||||
|  | from __future__ import annotations | ||||||
|  |  | ||||||
| import logging | import logging | ||||||
|  | import typing | ||||||
|  | from collections import Counter | ||||||
|  |  | ||||||
| logger = logging.getLogger("Hollow Knight") | logger = logging.getLogger("Hollow Knight") | ||||||
|  |  | ||||||
| from .Locations import lookup_name_to_id |  | ||||||
| from .Items import item_table, lookup_type_to_names | from .Items import item_table, lookup_type_to_names | ||||||
| from .Regions import create_regions | from .Regions import create_regions | ||||||
| from .Rules import set_rules | from .Rules import set_rules | ||||||
| from .Options import hollow_knight_options | from .Options import hollow_knight_options, hollow_knight_randomize_options | ||||||
|  | from .ExtractedData import locations, starts, multi_locations, location_to_region_lookup, \ | ||||||
|  |     event_names, item_effects, connectors, one_ways | ||||||
|  |  | ||||||
| from BaseClasses import Region, Entrance, Location, MultiWorld, Item, RegionType | from BaseClasses import Region, Entrance, Location, MultiWorld, Item, RegionType | ||||||
| from ..AutoWorld import World, LogicMixin | from ..AutoWorld import World, LogicMixin | ||||||
|  |  | ||||||
|  | white_palace_locations = { | ||||||
|  |     "Soul_Totem-Path_of_Pain_Below_Thornskip", | ||||||
|  |     "Soul_Totem-White_Palace_Final", | ||||||
|  |     "Lore_Tablet-Path_of_Pain_Entrance", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Left_of_Lever", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Hidden", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Entrance", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Final", | ||||||
|  |     "Soul_Totem-White_Palace_Entrance", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Below_Lever", | ||||||
|  |     "Lore_Tablet-Palace_Throne", | ||||||
|  |     "Soul_Totem-Path_of_Pain_Second", | ||||||
|  |     "Soul_Totem-White_Palace_Left", | ||||||
|  |     "Lore_Tablet-Palace_Workshop", | ||||||
|  |     "Soul_Totem-White_Palace_Hub", | ||||||
|  |     "Journal_Entry-Seal_of_Binding", | ||||||
|  |     "Soul_Totem-White_Palace_Right", | ||||||
|  |     "King_Fragment", | ||||||
|  |     # Events: | ||||||
|  |     "Palace_Entrance_Lantern_Lit", | ||||||
|  |     "Palace_Left_Lantern_Lit", | ||||||
|  |     "Palace_Right_Lantern_Lit", | ||||||
|  |     "Warp-Path_of_Pain_Complete", | ||||||
|  |     "Defeated_Path_of_Pain_Arena", | ||||||
|  |     "Palace_Atrium_Gates_Opened", | ||||||
|  |     "Completed_Path_of_Pain", | ||||||
|  |     "Warp-White_Palace_Atrium_to_Palace_Grounds", | ||||||
|  |     "Warp-White_Palace_Entrance_to_Palace_Grounds", | ||||||
|  |     # Event-Regions: | ||||||
|  |     "White_Palace_03_hub", | ||||||
|  |     "White_Palace_13", | ||||||
|  |     "White_Palace_01", | ||||||
|  |     # Event-Transitions: | ||||||
|  |     "White_Palace_12[bot1]", "White_Palace_12[bot1]", "White_Palace_03_hub[bot1]", "White_Palace_16[left2]", | ||||||
|  |     "White_Palace_16[left2]", "White_Palace_11[door2]", "White_Palace_11[door2]", "White_Palace_18[top1]", | ||||||
|  |     "White_Palace_18[top1]", "White_Palace_15[left1]", "White_Palace_15[left1]", "White_Palace_05[left2]", | ||||||
|  |     "White_Palace_05[left2]", "White_Palace_14[bot1]", "White_Palace_14[bot1]", "White_Palace_13[left2]", | ||||||
|  |     "White_Palace_13[left2]", "White_Palace_03_hub[left1]", "White_Palace_03_hub[left1]", "White_Palace_15[right2]", | ||||||
|  |     "White_Palace_15[right2]", "White_Palace_06[top1]", "White_Palace_06[top1]", "White_Palace_03_hub[bot1]", | ||||||
|  |     "White_Palace_08[right1]", "White_Palace_08[right1]", "White_Palace_03_hub[right1]", "White_Palace_03_hub[right1]", | ||||||
|  |     "White_Palace_01[right1]", "White_Palace_01[right1]", "White_Palace_08[left1]", "White_Palace_08[left1]", | ||||||
|  |     "White_Palace_19[left1]", "White_Palace_19[left1]", "White_Palace_04[right2]", "White_Palace_04[right2]", | ||||||
|  |     "White_Palace_01[left1]", "White_Palace_01[left1]", "White_Palace_17[right1]", "White_Palace_17[right1]", | ||||||
|  |     "White_Palace_07[bot1]", "White_Palace_07[bot1]", "White_Palace_20[bot1]", "White_Palace_20[bot1]", | ||||||
|  |     "White_Palace_03_hub[left2]", "White_Palace_03_hub[left2]", "White_Palace_18[right1]", "White_Palace_18[right1]", | ||||||
|  |     "White_Palace_05[right1]", "White_Palace_05[right1]", "White_Palace_17[bot1]", "White_Palace_17[bot1]", | ||||||
|  |     "White_Palace_09[right1]", "White_Palace_09[right1]", "White_Palace_16[left1]", "White_Palace_16[left1]", | ||||||
|  |     "White_Palace_13[left1]", "White_Palace_13[left1]", "White_Palace_06[bot1]", "White_Palace_06[bot1]", | ||||||
|  |     "White_Palace_15[right1]", "White_Palace_15[right1]", "White_Palace_06[left1]", "White_Palace_06[left1]", | ||||||
|  |     "White_Palace_05[right2]", "White_Palace_05[right2]", "White_Palace_04[top1]", "White_Palace_04[top1]", | ||||||
|  |     "White_Palace_19[top1]", "White_Palace_19[top1]", "White_Palace_14[right1]", "White_Palace_14[right1]", | ||||||
|  |     "White_Palace_03_hub[top1]", "White_Palace_03_hub[top1]", "Grubfather_2", "White_Palace_13[left3]", | ||||||
|  |     "White_Palace_13[left3]", "White_Palace_02[left1]", "White_Palace_02[left1]", "White_Palace_12[right1]", | ||||||
|  |     "White_Palace_12[right1]", "White_Palace_07[top1]", "White_Palace_07[top1]", "White_Palace_05[left1]", | ||||||
|  |     "White_Palace_05[left1]", "White_Palace_13[right1]", "White_Palace_13[right1]", "White_Palace_01[top1]", | ||||||
|  |     "White_Palace_01[top1]", | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| class HKWorld(World): | class HKWorld(World): | ||||||
|     game: str = "Hollow Knight" |     game: str = "Hollow Knight" | ||||||
|     options = hollow_knight_options |     options = hollow_knight_options | ||||||
|  |  | ||||||
|     item_name_to_id = {name: data.id for name, data in item_table.items() if data.type != "Event"} |     item_name_to_id = {name: data.id for name, data in item_table.items()} | ||||||
|     location_name_to_id = lookup_name_to_id |     location_name_to_id = {location_name: location_id for location_id, location_name in | ||||||
|  |                            enumerate(locations, start=0x1000000)} | ||||||
|  |  | ||||||
|     hidden = True |     hidden = True | ||||||
|  |     ranges: typing.Dict[str, typing.Tuple[int, int]] | ||||||
|  |     shops = {"Egg_Shop": "egg", "Grubfather": "grub", "Seer": "essence", "Salubra_(Requires_Charms)": "charm"} | ||||||
|  |     charm_costs: typing.List[int] | ||||||
|  |     data_version = 2 | ||||||
|  |  | ||||||
|  |     allow_white_palace = False | ||||||
|  |  | ||||||
|  |     def __init__(self, world, player): | ||||||
|  |         super(HKWorld, self).__init__(world, player) | ||||||
|  |         self.created_multi_locations: typing.Dict[str, int] = Counter() | ||||||
|  |         self.ranges = {} | ||||||
|  |  | ||||||
|  |     def generate_early(self): | ||||||
|  |         world = self.world | ||||||
|  |         self.charm_costs = world.random_charm_costs[self.player].get_costs(world.random) | ||||||
|  |         world.exclude_locations[self.player].value.update(white_palace_locations) | ||||||
|  |         world.local_items[self.player].value.add("Mimic_Grub") | ||||||
|  |         for vendor, unit in self.shops.items(): | ||||||
|  |             mini = getattr(world, f"minimum_{unit}_price")[self.player] | ||||||
|  |             maxi = getattr(world, f"maximum_{unit}_price")[self.player] | ||||||
|  |             # if minimum > maximum, set minimum to maximum | ||||||
|  |             mini.value = min(mini.value, maxi.value) | ||||||
|  |             self.ranges[unit] = mini.value, maxi.value | ||||||
|  |         world.push_precollected(HKItem(starts[world.start_location[self.player].current_key], | ||||||
|  |                                        True, None, "Event", self.player)) | ||||||
|  |  | ||||||
|  |     def create_regions(self): | ||||||
|  |         menu_region: Region = create_region(self.world, self.player, 'Menu') | ||||||
|  |         self.world.regions.append(menu_region) | ||||||
|  |  | ||||||
|     def generate_basic(self): |  | ||||||
|         # Link regions |         # Link regions | ||||||
|         self.world.get_entrance('Hollow Nest S&Q', self.player).connect(self.world.get_region('Hollow Nest', self.player)) |         for event_name in event_names: | ||||||
|  |             loc = HKLocation(self.player, event_name, None, menu_region) | ||||||
|  |             loc.place_locked_item(HKItem(event_name, | ||||||
|  |                                          self.allow_white_palace or event_name not in white_palace_locations, | ||||||
|  |                                          None, "Event", self.player)) | ||||||
|  |             menu_region.locations.append(loc) | ||||||
|  |         for entry_transition, exit_transition in connectors.items(): | ||||||
|  |             if exit_transition: | ||||||
|  |                 # if door logic fulfilled -> award vanilla target as event | ||||||
|  |                 loc = HKLocation(self.player, entry_transition, None, menu_region) | ||||||
|  |                 loc.place_locked_item(HKItem(exit_transition, | ||||||
|  |                                              self.allow_white_palace or exit_transition not in white_palace_locations, | ||||||
|  |                                              None, "Event", self.player)) | ||||||
|  |                 menu_region.locations.append(loc) | ||||||
|  |  | ||||||
|         # Generate item pool |     def create_items(self): | ||||||
|         pool = [] |         # Generate item pool and associated locations (paired in HK) | ||||||
|         for item_name, item_data in item_table.items(): |         pool: typing.List[HKItem] = [] | ||||||
|  |         geo_replace: typing.Set[str] = set() | ||||||
|  |         if self.world.RemoveSpellUpgrades[self.player]: | ||||||
|  |             geo_replace.add("Abyss_Shriek") | ||||||
|  |             geo_replace.add("Shade_Soul") | ||||||
|  |             geo_replace.add("Descending_Dark") | ||||||
|  |  | ||||||
|  |         for option_key, option in hollow_knight_randomize_options.items(): | ||||||
|  |             if getattr(self.world, option_key)[self.player]: | ||||||
|  |                 for item_name, location_name in zip(option.items, option.locations): | ||||||
|  |                     if item_name in geo_replace: | ||||||
|  |                         item_name = "Geo_Rock-Default" | ||||||
|  |                     if location_name in white_palace_locations: | ||||||
|  |                         self.create_location(location_name).place_locked_item(self.create_item(item_name)) | ||||||
|  |                     else: | ||||||
|  |                         self.create_location(location_name) | ||||||
|  |                         pool.append(self.create_item(item_name)) | ||||||
|  |             else: | ||||||
|  |                 for item_name, location_name in zip(option.items, option.locations): | ||||||
|                     item = self.create_item(item_name) |                     item = self.create_item(item_name) | ||||||
|  |                     if location_name == "Start": | ||||||
|             if item_data.type == "Event": |  | ||||||
|                 event_location = self.world.get_location(item_name, self.player) |  | ||||||
|                 self.world.push_item(event_location, item, collect=False) |  | ||||||
|                 event_location.event = True |  | ||||||
|                 event_location.locked = True |  | ||||||
|                 if item.name == "King's_Pass": |  | ||||||
|                         self.world.push_precollected(item) |                         self.world.push_precollected(item) | ||||||
|             elif item_data.type == "Cursed": |  | ||||||
|                 if self.world.CURSED[self.player]: |  | ||||||
|                     pool.append(item) |  | ||||||
|                     else: |                     else: | ||||||
|                     # fill Focus Location with Focus and add it to start inventory as well. |                         self.create_location(location_name).place_locked_item(item) | ||||||
|                     event_location = self.world.get_location(item_name, self.player) |         for i in range(self.world.egg_shop_slots[self.player].value): | ||||||
|                     self.world.push_item(event_location, item) |             self.create_location("Egg_Shop") | ||||||
|                     event_location.event = True |             pool.append(self.create_item("Geo_Rock-Default")) | ||||||
|                     event_location.locked = True |         if not self.allow_white_palace: | ||||||
|  |             loc = self.world.get_location("King_Fragment", self.player) | ||||||
|             elif item_data.type == "Fake": |             if loc.item and loc.item.name == loc.name: | ||||||
|                 pass |                 loc.item.advancement = False | ||||||
|             elif item_data.type in not_shufflable_types: |  | ||||||
|                 location = self.world.get_location(item_name, self.player) |  | ||||||
|                 self.world.push_item(location, item, collect=False) |  | ||||||
|                 location.event = item.advancement |  | ||||||
|                 location.locked = True |  | ||||||
|             else: |  | ||||||
|                 target = option_to_type_lookup[item.type] |  | ||||||
|                 shuffle_it = getattr(self.world, target) |  | ||||||
|                 if shuffle_it[self.player]: |  | ||||||
|                     pool.append(item) |  | ||||||
|                 else: |  | ||||||
|                     location = self.world.get_location(item_name, self.player) |  | ||||||
|                     self.world.push_item(location, item, collect=False) |  | ||||||
|                     location.event = item.advancement |  | ||||||
|                     location.locked = True |  | ||||||
|                     logger.debug(f"Placed {item_name} to vanilla for player {self.player}") |  | ||||||
|  |  | ||||||
|         self.world.itempool += pool |         self.world.itempool += pool | ||||||
|  |  | ||||||
|     def set_rules(self): |     def set_rules(self): | ||||||
|         set_rules(self.world, self.player) |         world = self.world | ||||||
|  |         player = self.player | ||||||
|     def create_regions(self): |         if world.logic[player] != 'nologic': | ||||||
|         create_regions(self.world, self.player) |             world.completion_condition[player] = lambda state: state.has('DREAMER', player, 3) | ||||||
|  |         set_rules(self) | ||||||
|  |  | ||||||
|     def fill_slot_data(self): |     def fill_slot_data(self): | ||||||
|         slot_data = {} |         slot_data = {} | ||||||
|  |  | ||||||
|  |         options = slot_data["options"] = {} | ||||||
|         for option_name in self.options: |         for option_name in self.options: | ||||||
|             option = getattr(self.world, option_name)[self.player] |             option = getattr(self.world, option_name)[self.player] | ||||||
|             slot_data[option_name] = int(option.value) |             options[option_name] = int(option.value) | ||||||
|  |  | ||||||
|  |         # 32 bit int | ||||||
|  |         slot_data["seed"] = self.world.slot_seeds[self.player].randint(-2147483647, 2147483646) | ||||||
|  |  | ||||||
|  |         for shop, unit in self.shops.items(): | ||||||
|  |             slot_data[f"{unit}_costs"] = { | ||||||
|  |                 f"{shop}_{i}": | ||||||
|  |                     self.world.get_location(f"{shop}_{i}", self.player).cost | ||||||
|  |                 for i in range(1, 1 + self.created_multi_locations[shop]) | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         slot_data["notch_costs"] = self.charm_costs | ||||||
|  |  | ||||||
|         return slot_data |         return slot_data | ||||||
|  |  | ||||||
|     def create_item(self, name: str) -> Item: |     def create_item(self, name: str) -> HKItem: | ||||||
|         item_data = item_table[name] |         item_data = item_table[name] | ||||||
|         return HKItem(name, item_data.advancement, item_data.id, item_data.type, self.player) |         return HKItem(name, item_data.advancement, item_data.id, item_data.type, self.player) | ||||||
|  |  | ||||||
|  |     def create_location(self, name: str) -> HKLocation: | ||||||
|  |         unit = self.shops.get(name, None) | ||||||
|  |         if unit: | ||||||
|  |             cost = self.world.random.randint(*self.ranges[unit]) | ||||||
|  |         else: | ||||||
|  |             cost = 0 | ||||||
|  |         if name in multi_locations: | ||||||
|  |             self.created_multi_locations[name] += 1 | ||||||
|  |             name += f"_{self.created_multi_locations[name]}" | ||||||
|  |  | ||||||
| def create_region(world: MultiWorld, player: int, name: str, locations=None, exits=None): |         region = self.world.get_region("Menu", self.player) | ||||||
|  |         loc = HKLocation(self.player, name, self.location_name_to_id[name], region) | ||||||
|  |         if unit: | ||||||
|  |             loc.unit = unit | ||||||
|  |             loc.cost = cost | ||||||
|  |         region.locations.append(loc) | ||||||
|  |         return loc | ||||||
|  |  | ||||||
|  |     def collect(self, state, item: HKItem) -> bool: | ||||||
|  |         change = super(HKWorld, self).collect(state, item) | ||||||
|  |  | ||||||
|  |         for effect_name, effect_value in item_effects.get(item.name, {}).items(): | ||||||
|  |             state.prog_items[effect_name, item.player] += effect_value | ||||||
|  |  | ||||||
|  |         return change | ||||||
|  |  | ||||||
|  |     def remove(self, state, item: HKItem) -> bool: | ||||||
|  |         change = super(HKWorld, self).remove(state, item) | ||||||
|  |  | ||||||
|  |         for effect_name, effect_value in item_effects.get(item.name, {}).items(): | ||||||
|  |             if state.prog_items[effect_name, item.player] == effect_value: | ||||||
|  |                 del state.prog_items[effect_name, item.player] | ||||||
|  |             state.prog_items[effect_name, item.player] -= effect_value | ||||||
|  |  | ||||||
|  |         return change | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def create_region(world: MultiWorld, player: int, name: str, location_names=None, exits=None) -> Region: | ||||||
|     ret = Region(name, RegionType.Generic, name, player) |     ret = Region(name, RegionType.Generic, name, player) | ||||||
|     ret.world = world |     ret.world = world | ||||||
|     if locations: |     if location_names: | ||||||
|         for location in locations: |         for location in location_names: | ||||||
|             loc_id = lookup_name_to_id.get(location, 0) |             loc_id = HKWorld.location_name_to_id.get(location, None) | ||||||
|             location = HKLocation(player, location, loc_id, ret) |             location = HKLocation(player, location, loc_id, ret) | ||||||
|             ret.locations.append(location) |             ret.locations.append(location) | ||||||
|     if exits: |     if exits: | ||||||
|         for exit in exits: |         for exit in exits: | ||||||
|             ret.exits.append(Entrance(player, exit, ret)) |             ret.exits.append(Entrance(player, exit, ret)) | ||||||
|  |  | ||||||
|     return ret |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| class HKLocation(Location): | class HKLocation(Location): | ||||||
|     game: str = "Hollow Knight" |     game: str = "Hollow Knight" | ||||||
|  |     cost: int = 0 | ||||||
|  |     unit: typing.Optional[str] = None | ||||||
|  |  | ||||||
|     def __init__(self, player: int, name: str, address=None, parent=None): |     def __init__(self, player: int, name: str, code=None, parent=None): | ||||||
|         super(HKLocation, self).__init__(player, name, address if address else None, parent) |         super(HKLocation, self).__init__(player, name, code if code else None, parent) | ||||||
|  |  | ||||||
|  |  | ||||||
| class HKItem(Item): | class HKItem(Item): | ||||||
| @@ -114,51 +265,20 @@ class HKItem(Item): | |||||||
|     def __init__(self, name, advancement, code, type, player: int = None): |     def __init__(self, name, advancement, code, type, player: int = None): | ||||||
|         super(HKItem, self).__init__(name, advancement, code if code else None, player) |         super(HKItem, self).__init__(name, advancement, code if code else None, player) | ||||||
|         self.type = type |         self.type = type | ||||||
|  |         if name == "Mimic_Grub": | ||||||
|  |             self.trap = True | ||||||
|  |  | ||||||
|  |  | ||||||
| not_shufflable_types = {"Essence_Boss"} | class HKLogicMixin(LogicMixin): | ||||||
|  |     world: MultiWorld | ||||||
|  |  | ||||||
| option_to_type_lookup = { |     def _hk_notches(self, player: int, *notches: int) -> int: | ||||||
|     "Root": "RandomizeWhisperingRoots", |         return sum(self.world.worlds[player].charm_costs[notch] for notch in notches) | ||||||
|     "Dreamer": "RandomizeDreamers", |  | ||||||
|     "Geo": "RandomizeGeoChests", |  | ||||||
|     "Skill": "RandomizeSkills", |  | ||||||
|     "Map": "RandomizeMaps", |  | ||||||
|     "Relic": "RandomizeRelics", |  | ||||||
|     "Charm": "RandomizeCharms", |  | ||||||
|     "Notch": "RandomizeCharmNotches", |  | ||||||
|     "Key": "RandomizeKeys", |  | ||||||
|     "Stag": "RandomizeStags", |  | ||||||
|     "Flame": "RandomizeFlames", |  | ||||||
|     "Grub": "RandomizeGrubs", |  | ||||||
|     "Cocoon": "RandomizeLifebloodCocoons", |  | ||||||
|     "Mask": "RandomizeMaskShards", |  | ||||||
|     "Ore": "RandomizePaleOre", |  | ||||||
|     "Egg": "RandomizeRancidEggs", |  | ||||||
|     "Vessel": "RandomizeVesselFragments", |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |     def _kh_option(self, player: int, option_name: str) -> int: | ||||||
|  |         if option_name == "RandomizeCharmNotches": | ||||||
|  |             return self.world.random_charm_costs[player] != -1 | ||||||
|  |         return getattr(self.world, option_name)[player].value | ||||||
|  |  | ||||||
| class HKLogic(LogicMixin): |     def _kh_start(self, player, start_location: str) -> bool: | ||||||
|     # these are all wip |         return self.world.start_location[player] == start_location | ||||||
|     def _hk_has_essence(self, player: int, count: int): |  | ||||||
|         return self.prog_items["Dream_Nail", player] |  | ||||||
|         # return self.prog_items["Essence", player] >= count |  | ||||||
|  |  | ||||||
|     def _hk_has_grubs(self, player: int, count: int): |  | ||||||
|         found = 0 |  | ||||||
|         for item_name in lookup_type_to_names["Grub"]: |  | ||||||
|             found += self.prog_items[item_name, player] |  | ||||||
|             if found >= count: |  | ||||||
|                 return True |  | ||||||
|  |  | ||||||
|         return False |  | ||||||
|  |  | ||||||
|     def _hk_has_flames(self, player: int, count: int): |  | ||||||
|         found = 0 |  | ||||||
|         for item_name in lookup_type_to_names["Flame"]: |  | ||||||
|             found += self.prog_items[item_name, player] |  | ||||||
|             if found >= count: |  | ||||||
|                 return True |  | ||||||
|  |  | ||||||
|         return False |  | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								worlds/hk/requirements.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								worlds/hk/requirements.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | astunparse>=1.6.3; python_version <= '3.8' | ||||||
		Reference in New Issue
	
	Block a user
	 Fabian Dill
					Fabian Dill