summary refs log tree commit diff stats
diff options
context:
space:
mode:
-rw-r--r--__init__.py23
-rw-r--r--items.py24
-rw-r--r--options.py14
-rw-r--r--player_logic.py19
4 files changed, 51 insertions, 29 deletions
diff --git a/__init__.py b/__init__.py index c92e530..b749418 100644 --- a/__init__.py +++ b/__init__.py
@@ -6,7 +6,7 @@ from logging import warning
6from BaseClasses import Item, ItemClassification, Tutorial 6from BaseClasses import Item, ItemClassification, Tutorial
7from worlds.AutoWorld import WebWorld, World 7from worlds.AutoWorld import WebWorld, World
8from .datatypes import Room, RoomEntrance 8from .datatypes import Room, RoomEntrance
9from .items import ALL_ITEM_TABLE, ITEMS_BY_GROUP, LingoItem 9from .items import ALL_ITEM_TABLE, ITEMS_BY_GROUP, TRAP_ITEMS, LingoItem
10from .locations import ALL_LOCATION_TABLE, LOCATIONS_BY_GROUP 10from .locations import ALL_LOCATION_TABLE, LOCATIONS_BY_GROUP
11from .options import LingoOptions 11from .options import LingoOptions
12from .player_logic import LingoPlayerLogic 12from .player_logic import LingoPlayerLogic
@@ -91,10 +91,23 @@ class LingoWorld(World):
91 pool.append(self.create_item("Puzzle Skip")) 91 pool.append(self.create_item("Puzzle Skip"))
92 92
93 if traps: 93 if traps:
94 traps_list = ["Slowness Trap", "Iceland Trap", "Atbash Trap"] 94 total_weight = sum(self.options.trap_weights.values())
95 95
96 for i in range(0, traps): 96 if total_weight == 0:
97 pool.append(self.create_item(traps_list[i % len(traps_list)])) 97 raise Exception("Sum of trap weights must be at least one.")
98
99 trap_counts = {name: int(weight * traps / total_weight)
100 for name, weight in self.options.trap_weights.items()}
101
102 trap_difference = traps - sum(trap_counts.values())
103 if trap_difference > 0:
104 allowed_traps = [name for name in TRAP_ITEMS if self.options.trap_weights[name] > 0]
105 for i in range(0, trap_difference):
106 trap_counts[allowed_traps[i % len(allowed_traps)]] += 1
107
108 for name, count in trap_counts.items():
109 for i in range(0, count):
110 pool.append(self.create_item(name))
98 111
99 self.multiworld.itempool += pool 112 self.multiworld.itempool += pool
100 113
diff --git a/items.py b/items.py index b9c4eb7..7c7928c 100644 --- a/items.py +++ b/items.py
@@ -1,13 +1,9 @@
1from typing import Dict, List, NamedTuple, Optional, TYPE_CHECKING 1from typing import Dict, List, NamedTuple, Optional, TYPE_CHECKING
2 2
3from BaseClasses import Item, ItemClassification 3from BaseClasses import Item, ItemClassification
4from .options import ShuffleDoors
5from .static_logic import DOORS_BY_ROOM, PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, get_door_group_item_id, \ 4from .static_logic import DOORS_BY_ROOM, PROGRESSION_BY_ROOM, PROGRESSIVE_ITEMS, get_door_group_item_id, \
6 get_door_item_id, get_progressive_item_id, get_special_item_id 5 get_door_item_id, get_progressive_item_id, get_special_item_id
7 6
8if TYPE_CHECKING:
9 from . import LingoWorld
10
11 7
12class ItemData(NamedTuple): 8class ItemData(NamedTuple):
13 """ 9 """
@@ -19,20 +15,6 @@ class ItemData(NamedTuple):
19 has_doors: bool 15 has_doors: bool
20 painting_ids: List[str] 16 painting_ids: List[str]
21 17
22 def should_include(self, world: "LingoWorld") -> bool:
23 if self.mode == "colors":
24 return world.options.shuffle_colors > 0
25 elif self.mode == "doors":
26 return world.options.shuffle_doors != ShuffleDoors.option_none
27 elif self.mode == "complex door":
28 return world.options.shuffle_doors == ShuffleDoors.option_complex
29 elif self.mode == "door group":
30 return world.options.shuffle_doors == ShuffleDoors.option_simple
31 elif self.mode == "special":
32 return False
33 else:
34 return True
35
36 18
37class LingoItem(Item): 19class LingoItem(Item):
38 """ 20 """
@@ -44,6 +26,8 @@ class LingoItem(Item):
44ALL_ITEM_TABLE: Dict[str, ItemData] = {} 26ALL_ITEM_TABLE: Dict[str, ItemData] = {}
45ITEMS_BY_GROUP: Dict[str, List[str]] = {} 27ITEMS_BY_GROUP: Dict[str, List[str]] = {}
46 28
29TRAP_ITEMS: List[str] = ["Slowness Trap", "Iceland Trap", "Atbash Trap"]
30
47 31
48def load_item_data(): 32def load_item_data():
49 global ALL_ITEM_TABLE, ITEMS_BY_GROUP 33 global ALL_ITEM_TABLE, ITEMS_BY_GROUP
@@ -87,9 +71,7 @@ def load_item_data():
87 "The Feeling of Being Lost": ItemClassification.filler, 71 "The Feeling of Being Lost": ItemClassification.filler,
88 "Wanderlust": ItemClassification.filler, 72 "Wanderlust": ItemClassification.filler,
89 "Empty White Hallways": ItemClassification.filler, 73 "Empty White Hallways": ItemClassification.filler,
90 "Slowness Trap": ItemClassification.trap, 74 **{trap_name: ItemClassification.trap for trap_name in TRAP_ITEMS},
91 "Iceland Trap": ItemClassification.trap,
92 "Atbash Trap": ItemClassification.trap,
93 "Puzzle Skip": ItemClassification.useful, 75 "Puzzle Skip": ItemClassification.useful,
94 } 76 }
95 77
diff --git a/options.py b/options.py index ed14264..293992a 100644 --- a/options.py +++ b/options.py
@@ -1,6 +1,9 @@
1from dataclasses import dataclass 1from dataclasses import dataclass
2 2
3from Options import Toggle, Choice, DefaultOnToggle, Range, PerGameCommonOptions, StartInventoryPool 3from schema import And, Schema
4
5from Options import Toggle, Choice, DefaultOnToggle, Range, PerGameCommonOptions, StartInventoryPool, OptionDict
6from worlds.lingo.items import TRAP_ITEMS
4 7
5 8
6class ShuffleDoors(Choice): 9class ShuffleDoors(Choice):
@@ -107,6 +110,14 @@ class TrapPercentage(Range):
107 default = 20 110 default = 20
108 111
109 112
113class TrapWeights(OptionDict):
114 """Specify the distribution of traps that should be placed into the pool.
115 If you don't want a specific type of trap, set the weight to zero."""
116 display_name = "Trap Weights"
117 schema = Schema({trap_name: And(int, lambda n: n >= 0) for trap_name in TRAP_ITEMS})
118 default = {trap_name: 1 for trap_name in TRAP_ITEMS}
119
120
110class PuzzleSkipPercentage(Range): 121class PuzzleSkipPercentage(Range):
111 """Replaces junk items with puzzle skips, at the specified rate.""" 122 """Replaces junk items with puzzle skips, at the specified rate."""
112 display_name = "Puzzle Skip Percentage" 123 display_name = "Puzzle Skip Percentage"
@@ -134,6 +145,7 @@ class LingoOptions(PerGameCommonOptions):
134 level_2_requirement: Level2Requirement 145 level_2_requirement: Level2Requirement
135 early_color_hallways: EarlyColorHallways 146 early_color_hallways: EarlyColorHallways
136 trap_percentage: TrapPercentage 147 trap_percentage: TrapPercentage
148 trap_weights: TrapWeights
137 puzzle_skip_percentage: PuzzleSkipPercentage 149 puzzle_skip_percentage: PuzzleSkipPercentage
138 death_link: DeathLink 150 death_link: DeathLink
139 start_inventory_from_pool: StartInventoryPool 151 start_inventory_from_pool: StartInventoryPool
diff --git a/player_logic.py b/player_logic.py index b3cefa5..966f5a1 100644 --- a/player_logic.py +++ b/player_logic.py
@@ -2,7 +2,7 @@ from enum import Enum
2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING 2from typing import Dict, List, NamedTuple, Optional, Set, Tuple, TYPE_CHECKING
3 3
4from .datatypes import Door, RoomAndDoor, RoomAndPanel 4from .datatypes import Door, RoomAndDoor, RoomAndPanel
5from .items import ALL_ITEM_TABLE 5from .items import ALL_ITEM_TABLE, ItemData
6from .locations import ALL_LOCATION_TABLE, LocationClassification 6from .locations import ALL_LOCATION_TABLE, LocationClassification
7from .options import LocationChecks, ShuffleDoors, VictoryCondition 7from .options import LocationChecks, ShuffleDoors, VictoryCondition
8from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \ 8from .static_logic import DOORS_BY_ROOM, PAINTINGS, PAINTING_ENTRANCES, PAINTING_EXITS, \
@@ -58,6 +58,21 @@ def should_split_progression(progression_name: str, world: "LingoWorld") -> Prog
58 return ProgressiveItemBehavior.PROGRESSIVE 58 return ProgressiveItemBehavior.PROGRESSIVE
59 59
60 60
61def should_include_item(item: ItemData, world: "LingoWorld") -> bool:
62 if item.mode == "colors":
63 return world.options.shuffle_colors > 0
64 elif item.mode == "doors":
65 return world.options.shuffle_doors != ShuffleDoors.option_none
66 elif item.mode == "complex door":
67 return world.options.shuffle_doors == ShuffleDoors.option_complex
68 elif item.mode == "door group":
69 return world.options.shuffle_doors == ShuffleDoors.option_simple
70 elif item.mode == "special":
71 return False
72 else:
73 return True
74
75
61class LingoPlayerLogic: 76class LingoPlayerLogic:
62 """ 77 """
63 Defines logic after a player's options have been applied 78 Defines logic after a player's options have been applied
@@ -212,7 +227,7 @@ class LingoPlayerLogic:
212 227
213 # Instantiate all real items. 228 # Instantiate all real items.
214 for name, item in ALL_ITEM_TABLE.items(): 229 for name, item in ALL_ITEM_TABLE.items():
215 if item.should_include(world): 230 if should_include_item(item, world):
216 self.real_items.append(name) 231 self.real_items.append(name)
217 232
218 # Calculate the requirements for the fake pilgrimage. 233 # Calculate the requirements for the fake pilgrimage.