mirror of
				https://github.com/MarioSpore/Grinch-AP.git
				synced 2025-10-21 20:21:32 -06:00 
			
		
		
		
	 703e3393a6
			
		
	
	703e3393a6
	
	
	
		
			
			* Add the yacht dice (from other git) world to the yacht dice fork * Update .gitignore * Removed zillion because it doesn't work * Update .gitignore * added zillion again... * Now you can have 0 extra fragments * Added alt categories, also options * Added item categories * Extra categories are now working! 🐶 * changed options and added exceptions * Testing if I change the generate.py * Revert "Testing if I change the generate.py" This reverts commit 7c2b3df6170dcf8d8f36a1de9fcbc9dccdec81f8. * ignore gitignore * Delete .gitignore * Update .gitignore * Update .gitignore * Update logic, added multiplicative categories * Changed difficulties * Update offline mode so that it works again * Adjusted difficulty * New version of the apworld, with 1000 as final score, always Will still need to check difficulty and weights of adding items. Website is not ready yet, so this version is not usable yet :) * Changed yaml and small bug fixes Fix when goal and max are same Options: changed chance to weight * no changes, just whitespaces * changed how logic works Now you put an array of mults and the cpu gets a couple of tries * Changed logic, tweaked a bit too * Preparation for 2.0 * logic tweak * Logic for alt categories properly now * Update setup_en.md * Update en_YachtDice.md * Improve performance of add_distributions * Formatting style * restore gitignore to APMW * Tweaked generation parameters and methods * Version 2.0.3 manual input option max score in logic always 2.0.3 faster gen * Comments and editing * Renamed setup guide * Improved create_items code * init of locations: remove self.event line * Moved setting early items to generate_early * Add my name to CODEOWNERS * Added Yacht Dice to the readme in list of games * Improve performance of Yacht Dice * newline * Improve typing * This is actually just slower lol * Update worlds/yachtdice/Items.py Co-authored-by: Exempt-Medic <60412657+Exempt-Medic@users.noreply.github.com> * Apply suggestions from code review Co-authored-by: Exempt-Medic <60412657+Exempt-Medic@users.noreply.github.com> * Update Options.py * Styling * finished text whichstory option * removed roll and rollfragments; not used * import; worlds not world :) * Option groups! * ruff styling, fix * ruff format styling! * styling and capitalization of options * small comment * Cleaned up the "state_is_a_list" a little bit * RUFF 🐶 * Changed filling the itempool for efficiency Now, we start with 17 extra items in the item pool, it's quite likely you need at least 17 items (~80%?). And then afterwards, we delete items if we overshoot the target of 1000, and add items if we haven't reached an achievable score of 1000 yet. Also, no need to recompute the entire logic when adding points. * 🐶 * Removed plando "fix" * Changed indent of score multiplier * faster location function * Comments to docstrings * fixed making location closest to goal_score be goal_score * options format * iterate keys and values of a dict together * small optimization ListState * faster collection of categories * return arguments instead of making a list (will 🐶 later) * Instead of turning it into a tuple, you can just make a tuple literal * remove .keys() * change .random and used enumerate * some readability improvements * Remove location "0", we don't use that one * Remove lookup_id_to_name entirely I for sure don't use it, and as far as I know it's not one of the mandatory functions for AP, these are item_name_to_id and location_name_to_id. * .append instead of += for single items, percentile function changed Also an extra comment for location ids. * remove ) too many * Removed sorted from category list * Hash categories (which makes it slower :( ) Maybe I messed up or misunderstood... I'll revert this right away since it is 2x slower, probably because of sorted instead of sort? * Revert "Hash categories (which makes it slower :( )" This reverts commit 34f2c1aed8c8813b2d9c58896650b82a810d3578. * temporary push: 40% faster generation test Small changes in logic make the generation 40% faster. I'll have to think about how big the changes are. I suspect they are rather limited. If this is the way to go, I'll remove the temp file and redo the YachtWeights file, I'll remove the functions there and just put the new weights here. * Add Points item category * Reverse changes of bad idea :) * ruff 🐶 * Use numpy and pmf function to speed up gen Numpy has a built-in way to sum probability mass functions (pmf). This shaves of 60% of the generation time :D * Revert "Use numpy and pmf function to speed up gen" This reverts commit 9290191cb323ae92321d6c2cfcfe8c27370f439b. * Step inbetween to change the weights * Changed the weights to make it faster 135 -> 81 seconds on 100 random yamls * Adjusted max_dist, split dice_simulation function * Removed nonlocal and pass arguments instead * Change "weight-lists" to Dict[str, float] * Removed the return from ini_locations. Also added explanations to cat_weights * Choice options; dont'use .value (will ruff later) * Only put important options in slotdata * 🐶 * Add Dict import * Split the cache per player, limit size to 400. * 🐶 * added , because of style * Update apworld version to 2.0.6 2.0.5 is the apworld I released on github to be tested I never separately released 2.0.4. * Multiple smaller code improvements - changed names in YachtWeights so we don't need to translate them in Rules anymore - we now remember which categories are present in the game, and also put this in slotdata. This we do because only one of two categories is present in a game. If for some reason both are present (plando/getitem/startinventory), we now know which category to ignore - * 🐶 ruff * Mostly minimize_extra_items improvements - Change logic, generation is now even faster (0.6s per default yaml). - Made the option 'minimize_extra_items' do a lot more, hopefully this makes the impact of Yacht Dice a little bit less, if you want that. Here's what is also does now: - you start with 2 dice and 2 rolls - there will be less locations/items at the start of you game * ruff 🐶 * Removed printing options * Reworded some option descriptions * Yacht Dice: setup: change release-link to latest On the installation page, link to the latest release, instead of the page with all releases * Several fixes and changes -change apworld version -Removed the extra roll (this was not intended) -change extra_points_added to a mutable list to that it actually does something -removed variables multipliers_added and items_added -Rules, don't order by quantity, just by mean_score -Changed the weights in general to make it faster * 🐶 * Revert setup to what it was (latest, without S) * remove temp weights file, shouldn't be here * Made sure that there is not too many step score multipliers. Too many step score multipliers lead to gen fails too, probably because you need many categories for them to actually help a lot. So it's hard to use them at the start of the game. * add filler item name * Textual fixes and changes * Remove Victory item and use event instead. * Revert "Remove Victory item and use event instead." This reverts commit c2f7d674d392a3acbc1db8614411164ba3b28bff. * Revert "Textual fixes and changes" This reverts commit e9432f92454979fcd5a31f8517586585362a7ab7. * Remove Victory item and make it an event instead * Yacht Dice logic fix, no decreasing score when obtain item take 2 * Logic fix: Revert max_tries and mults, change ordering * Remove spaces :^) * Updated weights that are stochastically ordered by dice/roll In the trimming of the weights, sometimes it having 4 rolls would be better than having 5 rolls. I did a check that this does not happen for any dice increment or roll increment * Swap for-loops to increase performance This method is faster if the first for-loop contains fewer items. Since the function is called with, typically, `dist2` having less items, let's loop over `dist2` first. This makes the entire program 10% faster. * Remove options with 0 chance from list --------- Co-authored-by: NewSoupVi <57900059+NewSoupVi@users.noreply.github.com> Co-authored-by: Exempt-Medic <60412657+Exempt-Medic@users.noreply.github.com>
		
			
				
	
	
		
			531 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			531 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import math
 | |
| from typing import Dict
 | |
| 
 | |
| from BaseClasses import CollectionState, Entrance, Item, ItemClassification, Location, Region, Tutorial
 | |
| 
 | |
| from worlds.AutoWorld import WebWorld, World
 | |
| 
 | |
| from .Items import YachtDiceItem, item_groups, item_table
 | |
| from .Locations import YachtDiceLocation, all_locations, ini_locations
 | |
| from .Options import (
 | |
|     AddExtraPoints,
 | |
|     AddStoryChapters,
 | |
|     GameDifficulty,
 | |
|     MinimalNumberOfDiceAndRolls,
 | |
|     MinimizeExtraItems,
 | |
|     PointsSize,
 | |
|     YachtDiceOptions,
 | |
|     yd_option_groups,
 | |
| )
 | |
| from .Rules import dice_simulation_fill_pool, set_yacht_completion_rules, set_yacht_rules
 | |
| 
 | |
| 
 | |
| class YachtDiceWeb(WebWorld):
 | |
|     tutorials = [
 | |
|         Tutorial(
 | |
|             "Multiworld Setup Guide",
 | |
|             "A guide to setting up Yacht Dice. This guide covers single-player, multiworld, and website.",
 | |
|             "English",
 | |
|             "setup_en.md",
 | |
|             "setup/en",
 | |
|             ["Spineraks"],
 | |
|         )
 | |
|     ]
 | |
| 
 | |
|     option_groups = yd_option_groups
 | |
| 
 | |
| 
 | |
| class YachtDiceWorld(World):
 | |
|     """
 | |
|     Yacht Dice is a straightforward game, custom-made for Archipelago,
 | |
|     where you cast your dice to chart a course for high scores,
 | |
|     unlocking valuable treasures along the way.
 | |
|     Discover more dice, extra rolls, multipliers,
 | |
|     and unlockable categories to navigate the depths of the game.
 | |
|     Roll your way to victory by reaching the target score!
 | |
|     """
 | |
| 
 | |
|     game: str = "Yacht Dice"
 | |
|     options_dataclass = YachtDiceOptions
 | |
| 
 | |
|     web = YachtDiceWeb()
 | |
| 
 | |
|     item_name_to_id = {name: data.code for name, data in item_table.items()}
 | |
| 
 | |
|     location_name_to_id = {name: data.id for name, data in all_locations.items()}
 | |
| 
 | |
|     item_name_groups = item_groups
 | |
| 
 | |
|     ap_world_version = "2.1.4"
 | |
| 
 | |
|     def _get_yachtdice_data(self):
 | |
|         return {
 | |
|             # "world_seed": self.multiworld.per_slot_randoms[self.player].getrandbits(32),
 | |
|             "seed_name": self.multiworld.seed_name,
 | |
|             "player_name": self.multiworld.get_player_name(self.player),
 | |
|             "player_id": self.player,
 | |
|             "race": self.multiworld.is_race,
 | |
|         }
 | |
| 
 | |
|     def generate_early(self):
 | |
|         """
 | |
|         In generate early, we fill the item-pool, then determine the number of locations, and add filler items.
 | |
|         """
 | |
|         self.itempool = []
 | |
|         self.precollected = []
 | |
| 
 | |
|         # number of dice and rolls in the pull
 | |
|         opt_dice_and_rolls = self.options.minimal_number_of_dice_and_rolls
 | |
| 
 | |
|         if opt_dice_and_rolls == MinimalNumberOfDiceAndRolls.option_5_dice_and_3_rolls:
 | |
|             num_of_dice = 5
 | |
|             num_of_rolls = 3
 | |
|         elif opt_dice_and_rolls == MinimalNumberOfDiceAndRolls.option_5_dice_and_5_rolls:
 | |
|             num_of_dice = 5
 | |
|             num_of_rolls = 5
 | |
|         elif opt_dice_and_rolls == MinimalNumberOfDiceAndRolls.option_6_dice_and_4_rolls:
 | |
|             num_of_dice = 6
 | |
|             num_of_rolls = 4
 | |
|         elif opt_dice_and_rolls == MinimalNumberOfDiceAndRolls.option_7_dice_and_3_rolls:
 | |
|             num_of_dice = 7
 | |
|             num_of_rolls = 3
 | |
|         elif opt_dice_and_rolls == MinimalNumberOfDiceAndRolls.option_8_dice_and_2_rolls:
 | |
|             num_of_dice = 8
 | |
|             num_of_rolls = 2
 | |
|         else:
 | |
|             raise Exception(f"[Yacht Dice] Unknown MinimalNumberOfDiceAndRolls options {opt_dice_and_rolls}")
 | |
| 
 | |
|         # amount of dice and roll fragments needed to get a dice or roll
 | |
|         self.frags_per_dice = self.options.number_of_dice_fragments_per_dice.value
 | |
|         self.frags_per_roll = self.options.number_of_roll_fragments_per_roll.value
 | |
| 
 | |
|         if self.options.minimize_extra_items == MinimizeExtraItems.option_yes_please:
 | |
|             self.frags_per_dice = min(self.frags_per_dice, 2)
 | |
|             self.frags_per_roll = min(self.frags_per_roll, 2)
 | |
| 
 | |
|         # set difficulty
 | |
|         diff_value = self.options.game_difficulty
 | |
|         if diff_value == GameDifficulty.option_easy:
 | |
|             self.difficulty = 1
 | |
|         elif diff_value == GameDifficulty.option_medium:
 | |
|             self.difficulty = 2
 | |
|         elif diff_value == GameDifficulty.option_hard:
 | |
|             self.difficulty = 3
 | |
|         elif diff_value == GameDifficulty.option_extreme:
 | |
|             self.difficulty = 4
 | |
|         else:
 | |
|             raise Exception(f"[Yacht Dice] Unknown GameDifficulty options {diff_value}")
 | |
| 
 | |
|         # Create a list with the specified number of 1s
 | |
|         num_ones = self.options.alternative_categories.value
 | |
|         categorylist = [1] * num_ones + [0] * (16 - num_ones)
 | |
| 
 | |
|         # Shuffle the list to randomize the order
 | |
|         self.random.shuffle(categorylist)
 | |
| 
 | |
|         # A list of all possible categories.
 | |
|         # Every entry in the list has two categories, one 'default' category and one 'alt'.
 | |
|         # You get either of the two for every entry, so a total of 16 unique categories.
 | |
|         all_categories = [
 | |
|             ["Category Choice", "Category Double Threes and Fours"],
 | |
|             ["Category Inverse Choice", "Category Quadruple Ones and Twos"],
 | |
|             ["Category Ones", "Category Distincts"],
 | |
|             ["Category Twos", "Category Two times Ones"],
 | |
|             ["Category Threes", "Category Half of Sixes"],
 | |
|             ["Category Fours", "Category Twos and Threes"],
 | |
|             ["Category Fives", "Category Sum of Odds"],
 | |
|             ["Category Sixes", "Category Sum of Evens"],
 | |
|             ["Category Pair", "Category Micro Straight"],
 | |
|             ["Category Three of a Kind", "Category Three Odds"],
 | |
|             ["Category Four of a Kind", "Category 1-2-1 Consecutive"],
 | |
|             ["Category Tiny Straight", "Category Three Distinct Dice"],
 | |
|             ["Category Small Straight", "Category Two Pair"],
 | |
|             ["Category Large Straight", "Category 2-1-2 Consecutive"],
 | |
|             ["Category Full House", "Category Five Distinct Dice"],
 | |
|             ["Category Yacht", "Category 4&5 Full House"],
 | |
|         ]
 | |
| 
 | |
|         # categories used in this game.
 | |
|         self.possible_categories = []
 | |
| 
 | |
|         for index, cats in enumerate(all_categories):
 | |
|             self.possible_categories.append(cats[categorylist[index]])
 | |
| 
 | |
|             # Add Choice and Inverse choice (or their alts) to the precollected list.
 | |
|             if index == 0 or index == 1:
 | |
|                 self.precollected.append(cats[categorylist[index]])
 | |
|             else:
 | |
|                 self.itempool.append(cats[categorylist[index]])
 | |
| 
 | |
|         # Also start with one Roll and one Dice
 | |
|         self.precollected.append("Dice")
 | |
|         num_of_dice_to_add = num_of_dice - 1
 | |
|         self.precollected.append("Roll")
 | |
|         num_of_rolls_to_add = num_of_rolls - 1
 | |
| 
 | |
|         self.skip_early_locations = False
 | |
|         if self.options.minimize_extra_items == MinimizeExtraItems.option_yes_please:
 | |
|             self.precollected.append("Dice")
 | |
|             num_of_dice_to_add -= 1
 | |
|             self.precollected.append("Roll")
 | |
|             num_of_rolls_to_add -= 1
 | |
|             self.skip_early_locations = True
 | |
| 
 | |
|         if num_of_dice_to_add > 0:
 | |
|             self.itempool.append("Dice")
 | |
|             num_of_dice_to_add -= 1
 | |
|         if num_of_rolls_to_add > 0:
 | |
|             self.itempool.append("Roll")
 | |
|             num_of_rolls_to_add -= 1
 | |
| 
 | |
|         # if one fragment per dice, just add "Dice" objects
 | |
|         if num_of_dice_to_add > 0:
 | |
|             if self.frags_per_dice == 1:
 | |
|                 self.itempool += ["Dice"] * num_of_dice_to_add  # minus one because one is in start inventory
 | |
|             else:
 | |
|                 self.itempool += ["Dice Fragment"] * (self.frags_per_dice * num_of_dice_to_add)
 | |
| 
 | |
|         # if one fragment per roll, just add "Roll" objects
 | |
|         if num_of_rolls_to_add > 0:
 | |
|             if self.frags_per_roll == 1:
 | |
|                 self.itempool += ["Roll"] * num_of_rolls_to_add  # minus one because one is in start inventory
 | |
|             else:
 | |
|                 self.itempool += ["Roll Fragment"] * (self.frags_per_roll * num_of_rolls_to_add)
 | |
| 
 | |
|         already_items = len(self.itempool)
 | |
| 
 | |
|         # Yacht Dice needs extra filler items so it doesn't get stuck in generation.
 | |
|         # For now, we calculate the number of extra items we'll need later.
 | |
|         if self.options.minimize_extra_items == MinimizeExtraItems.option_yes_please:
 | |
|             extra_percentage = max(0.1, 0.8 - self.multiworld.players / 10)
 | |
|         elif self.options.minimize_extra_items == MinimizeExtraItems.option_no_dont:
 | |
|             extra_percentage = 0.72
 | |
|         else:
 | |
|             raise Exception(f"[Yacht Dice] Unknown MinimizeExtraItems options {self.options.minimize_extra_items}")
 | |
|         extra_locations_needed = max(10, math.ceil(already_items * extra_percentage))
 | |
| 
 | |
|         # max score is the value of the last check. Goal score is the score needed to 'finish' the game
 | |
|         self.max_score = self.options.score_for_last_check.value
 | |
|         self.goal_score = min(self.max_score, self.options.score_for_goal.value)
 | |
| 
 | |
|         # Yacht Dice adds items into the pool until a score of at least 1000 is reached.
 | |
|         # the yaml contains weights, which determine how likely it is that specific items get added.
 | |
|         # If all weights are 0, some of them will be made to be non-zero later.
 | |
|         weights: Dict[str, float] = {
 | |
|             "Dice": self.options.weight_of_dice.value,
 | |
|             "Roll": self.options.weight_of_roll.value,
 | |
|             "Fixed Score Multiplier": self.options.weight_of_fixed_score_multiplier.value,
 | |
|             "Step Score Multiplier": self.options.weight_of_step_score_multiplier.value,
 | |
|             "Double category": self.options.weight_of_double_category.value,
 | |
|             "Points": self.options.weight_of_points.value,
 | |
|         }
 | |
| 
 | |
|         # if the player wants extra rolls or dice, fill the pool with fragments until close to an extra roll/dice
 | |
|         if weights["Dice"] > 0 and self.frags_per_dice > 1:
 | |
|             self.itempool += ["Dice Fragment"] * (self.frags_per_dice - 1)
 | |
|         if weights["Roll"] > 0 and self.frags_per_roll > 1:
 | |
|             self.itempool += ["Roll Fragment"] * (self.frags_per_roll - 1)
 | |
| 
 | |
|         # calibrate the weights, since the impact of each of the items is different
 | |
|         weights["Dice"] = weights["Dice"] / 5 * self.frags_per_dice
 | |
|         weights["Roll"] = weights["Roll"] / 5 * self.frags_per_roll
 | |
| 
 | |
|         extra_points_added = [0]  # make it a mutible type so we can change the value in the function
 | |
|         step_score_multipliers_added = [0]
 | |
| 
 | |
|         def get_item_to_add(weights, extra_points_added, step_score_multipliers_added):
 | |
|             all_items = self.itempool + self.precollected
 | |
|             dice_fragments_in_pool = all_items.count("Dice") * self.frags_per_dice + all_items.count("Dice Fragment")
 | |
|             if dice_fragments_in_pool + 1 >= 9 * self.frags_per_dice:
 | |
|                 weights["Dice"] = 0  # don't allow >=9 dice
 | |
|             roll_fragments_in_pool = all_items.count("Roll") * self.frags_per_roll + all_items.count("Roll Fragment")
 | |
|             if roll_fragments_in_pool + 1 >= 6 * self.frags_per_roll:
 | |
|                 weights["Roll"] = 0  # don't allow >= 6 rolls
 | |
| 
 | |
|             # Don't allow too many extra points
 | |
|             if extra_points_added[0] > 400:
 | |
|                 weights["Points"] = 0
 | |
| 
 | |
|             if step_score_multipliers_added[0] > 10:
 | |
|                 weights["Step Score Multiplier"] = 0
 | |
| 
 | |
|             # if all weights are zero, allow to add fixed score multiplier, double category, points.
 | |
|             if sum(weights.values()) == 0:
 | |
|                 weights["Fixed Score Multiplier"] = 1
 | |
|                 weights["Double category"] = 1
 | |
|                 if extra_points_added[0] <= 400:
 | |
|                     weights["Points"] = 1
 | |
| 
 | |
|             # Next, add the appropriate item. We'll slightly alter weights to avoid too many of the same item
 | |
|             which_item_to_add = self.random.choices(list(weights.keys()), weights=list(weights.values()))[0]
 | |
| 
 | |
|             if which_item_to_add == "Dice":
 | |
|                 weights["Dice"] /= 1 + self.frags_per_dice
 | |
|                 return "Dice" if self.frags_per_dice == 1 else "Dice Fragment"
 | |
|             elif which_item_to_add == "Roll":
 | |
|                 weights["Roll"] /= 1 + self.frags_per_roll
 | |
|                 return "Roll" if self.frags_per_roll == 1 else "Roll Fragment"
 | |
|             elif which_item_to_add == "Fixed Score Multiplier":
 | |
|                 weights["Fixed Score Multiplier"] /= 1.05
 | |
|                 return "Fixed Score Multiplier"
 | |
|             elif which_item_to_add == "Step Score Multiplier":
 | |
|                 weights["Step Score Multiplier"] /= 1.1
 | |
|                 step_score_multipliers_added[0] += 1
 | |
|                 return "Step Score Multiplier"
 | |
|             elif which_item_to_add == "Double category":
 | |
|                 # Below entries are the weights to add each category.
 | |
|                 # Prefer to add choice or number categories, because the other categories are too "all or nothing",
 | |
|                 # which often don't give any points, until you get overpowered, and then they give all points.
 | |
|                 cat_weights = [2, 2, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1]
 | |
|                 weights["Double category"] /= 1.1
 | |
|                 return self.random.choices(self.possible_categories, weights=cat_weights)[0]
 | |
|             elif which_item_to_add == "Points":
 | |
|                 score_dist = self.options.points_size
 | |
|                 probs = {"1 Point": 1, "10 Points": 0, "100 Points": 0}
 | |
|                 if score_dist == PointsSize.option_small:
 | |
|                     probs = {"1 Point": 0.9, "10 Points": 0.1, "100 Points": 0}
 | |
|                 elif score_dist == PointsSize.option_medium:
 | |
|                     probs = {"1 Point": 0, "10 Points": 1, "100 Points": 0}
 | |
|                 elif score_dist == PointsSize.option_large:
 | |
|                     probs = {"1 Point": 0, "10 Points": 0.3, "100 Points": 0.7}
 | |
|                 elif score_dist == PointsSize.option_mix:
 | |
|                     probs = {"1 Point": 0.3, "10 Points": 0.4, "100 Points": 0.3}
 | |
|                 else:
 | |
|                     raise Exception(f"[Yacht Dice] Unknown PointsSize options {score_dist}")
 | |
|                 choice = self.random.choices(list(probs.keys()), weights=list(probs.values()))[0]
 | |
|                 if choice == "1 Point":
 | |
|                     weights["Points"] /= 1.01
 | |
|                     extra_points_added[0] += 1
 | |
|                     return "1 Point"
 | |
|                 elif choice == "10 Points":
 | |
|                     weights["Points"] /= 1.1
 | |
|                     extra_points_added[0] += 10
 | |
|                     return "10 Points"
 | |
|                 elif choice == "100 Points":
 | |
|                     weights["Points"] /= 2
 | |
|                     extra_points_added[0] += 100
 | |
|                     return "100 Points"
 | |
|                 else:
 | |
|                     raise Exception("Unknown point value (Yacht Dice)")
 | |
|             else:
 | |
|                 raise Exception(f"Invalid index when adding new items in Yacht Dice: {which_item_to_add}")
 | |
| 
 | |
|         # adding 17 items as a start seems like the smartest way to get close to 1000 points
 | |
|         for _ in range(17):
 | |
|             self.itempool.append(get_item_to_add(weights, extra_points_added, step_score_multipliers_added))
 | |
| 
 | |
|         score_in_logic = dice_simulation_fill_pool(
 | |
|             self.itempool + self.precollected,
 | |
|             self.frags_per_dice,
 | |
|             self.frags_per_roll,
 | |
|             self.possible_categories,
 | |
|             self.difficulty,
 | |
|             self.player,
 | |
|         )
 | |
| 
 | |
|         # if we overshoot, remove items until you get below 1000, then return the last removed item
 | |
|         if score_in_logic > 1000:
 | |
|             removed_item = ""
 | |
|             while score_in_logic > 1000:
 | |
|                 removed_item = self.itempool.pop()
 | |
|                 score_in_logic = dice_simulation_fill_pool(
 | |
|                     self.itempool + self.precollected,
 | |
|                     self.frags_per_dice,
 | |
|                     self.frags_per_roll,
 | |
|                     self.possible_categories,
 | |
|                     self.difficulty,
 | |
|                     self.player,
 | |
|                 )
 | |
|             self.itempool.append(removed_item)
 | |
|         else:
 | |
|             # Keep adding items until a score of 1000 is in logic
 | |
|             while score_in_logic < 1000:
 | |
|                 item_to_add = get_item_to_add(weights, extra_points_added, step_score_multipliers_added)
 | |
|                 self.itempool.append(item_to_add)
 | |
|                 if item_to_add == "1 Point":
 | |
|                     score_in_logic += 1
 | |
|                 elif item_to_add == "10 Points":
 | |
|                     score_in_logic += 10
 | |
|                 elif item_to_add == "100 Points":
 | |
|                     score_in_logic += 100
 | |
|                 else:
 | |
|                     score_in_logic = dice_simulation_fill_pool(
 | |
|                         self.itempool + self.precollected,
 | |
|                         self.frags_per_dice,
 | |
|                         self.frags_per_roll,
 | |
|                         self.possible_categories,
 | |
|                         self.difficulty,
 | |
|                         self.player,
 | |
|                     )
 | |
| 
 | |
|         # count the number of locations in the game.
 | |
|         already_items = len(self.itempool) + 1  # +1 because of Victory item
 | |
| 
 | |
|         # We need to add more filler/useful items if there are many items in the pool to guarantee successful generation
 | |
|         extra_locations_needed += (already_items - 45) // 15
 | |
|         self.number_of_locations = already_items + extra_locations_needed
 | |
| 
 | |
|         # From here, we will count the number of items in the self.itempool, and add useful/filler items to the pool,
 | |
|         # making sure not to exceed the number of locations.
 | |
| 
 | |
|         # first, we flood the entire pool with extra points (useful), if that setting is chosen.
 | |
|         if self.options.add_bonus_points == AddExtraPoints.option_all_of_it:  # all of the extra points
 | |
|             already_items = len(self.itempool) + 1
 | |
|             self.itempool += ["Bonus Point"] * min(self.number_of_locations - already_items, 100)
 | |
| 
 | |
|         # second, we flood the entire pool with story chapters (filler), if that setting is chosen.
 | |
|         if self.options.add_story_chapters == AddStoryChapters.option_all_of_it:  # all of the story chapters
 | |
|             already_items = len(self.itempool) + 1
 | |
|             number_of_items = min(self.number_of_locations - already_items, 100)
 | |
|             number_of_items = (number_of_items // 10) * 10  # story chapters always come in multiples of 10
 | |
|             self.itempool += ["Story Chapter"] * number_of_items
 | |
| 
 | |
|         # add some extra points (useful)
 | |
|         if self.options.add_bonus_points == AddExtraPoints.option_sure:  # add extra points if wanted
 | |
|             already_items = len(self.itempool) + 1
 | |
|             self.itempool += ["Bonus Point"] * min(self.number_of_locations - already_items, 10)
 | |
| 
 | |
|         # add some story chapters (filler)
 | |
|         if self.options.add_story_chapters == AddStoryChapters.option_sure:  # add extra points if wanted
 | |
|             already_items = len(self.itempool) + 1
 | |
|             if self.number_of_locations - already_items >= 10:
 | |
|                 self.itempool += ["Story Chapter"] * 10
 | |
| 
 | |
|         # add some more extra points if there is still room
 | |
|         if self.options.add_bonus_points == AddExtraPoints.option_sure:
 | |
|             already_items = len(self.itempool) + 1
 | |
|             self.itempool += ["Bonus Point"] * min(self.number_of_locations - already_items, 10)
 | |
| 
 | |
|         # add some encouragements filler-items if there is still room
 | |
|         already_items = len(self.itempool) + 1
 | |
|         self.itempool += ["Encouragement"] * min(self.number_of_locations - already_items, 5)
 | |
| 
 | |
|         # add some fun facts filler-items if there is still room
 | |
|         already_items = len(self.itempool) + 1
 | |
|         self.itempool += ["Fun Fact"] * min(self.number_of_locations - already_items, 5)
 | |
| 
 | |
|         # finally, add some "Good RNG" and "Bad RNG" items to complete the item pool
 | |
|         # these items are filler and do not do anything.
 | |
| 
 | |
|         # probability of Good and Bad rng, based on difficulty for fun :)
 | |
| 
 | |
|         p = 1.1 - 0.25 * self.difficulty
 | |
|         already_items = len(self.itempool) + 1
 | |
|         self.itempool += self.random.choices(
 | |
|             ["Good RNG", "Bad RNG"], weights=[p, 1 - p], k=self.number_of_locations - already_items
 | |
|         )
 | |
| 
 | |
|         # we are done adding items. Now because of the last step, number of items should be number of locations
 | |
|         already_items = len(self.itempool) + 1
 | |
|         if already_items != self.number_of_locations:
 | |
|             raise Exception(
 | |
|                 f"[Yacht Dice] Number in self.itempool is not number of locations "
 | |
|                 f"{already_items} {self.number_of_locations}."
 | |
|             )
 | |
| 
 | |
|         # add precollected items using push_precollected. Items in self.itempool get created in create_items
 | |
|         for item in self.precollected:
 | |
|             self.multiworld.push_precollected(self.create_item(item))
 | |
| 
 | |
|         # make sure one dice and one roll is early, so that you will have 2 dice and 2 rolls soon
 | |
|         self.multiworld.early_items[self.player]["Dice"] = 1
 | |
|         self.multiworld.early_items[self.player]["Roll"] = 1
 | |
| 
 | |
|     def create_items(self):
 | |
|         self.multiworld.itempool += [self.create_item(name) for name in self.itempool]
 | |
| 
 | |
|     def create_regions(self):
 | |
|         # call the ini_locations function, that generates locations based on the inputs.
 | |
|         location_table = ini_locations(
 | |
|             self.goal_score,
 | |
|             self.max_score,
 | |
|             self.number_of_locations,
 | |
|             self.difficulty,
 | |
|             self.skip_early_locations,
 | |
|             self.multiworld.players,
 | |
|         )
 | |
| 
 | |
|         # simple menu-board construction
 | |
|         menu = Region("Menu", self.player, self.multiworld)
 | |
|         board = Region("Board", self.player, self.multiworld)
 | |
| 
 | |
|         # add locations to board, one for every location in the location_table
 | |
|         board.locations = [
 | |
|             YachtDiceLocation(self.player, loc_name, loc_data.score, loc_data.id, board)
 | |
|             for loc_name, loc_data in location_table.items()
 | |
|             if loc_data.region == board.name
 | |
|         ]
 | |
| 
 | |
|         # Change the victory location to an event and place the Victory item there.
 | |
|         victory_location_name = f"{self.goal_score} score"
 | |
|         self.get_location(victory_location_name).address = None
 | |
|         self.get_location(victory_location_name).place_locked_item(
 | |
|             Item("Victory", ItemClassification.progression, None, self.player)
 | |
|         )
 | |
| 
 | |
|         # add the regions
 | |
|         connection = Entrance(self.player, "New Board", menu)
 | |
|         menu.exits.append(connection)
 | |
|         connection.connect(board)
 | |
|         self.multiworld.regions += [menu, board]
 | |
| 
 | |
|     def get_filler_item_name(self) -> str:
 | |
|         return "Good RNG"
 | |
| 
 | |
|     def set_rules(self):
 | |
|         """
 | |
|         set rules per location, and add the rule for beating the game
 | |
|         """
 | |
|         set_yacht_rules(
 | |
|             self.multiworld,
 | |
|             self.player,
 | |
|             self.frags_per_dice,
 | |
|             self.frags_per_roll,
 | |
|             self.possible_categories,
 | |
|             self.difficulty,
 | |
|         )
 | |
|         set_yacht_completion_rules(self.multiworld, self.player)
 | |
| 
 | |
|     def fill_slot_data(self):
 | |
|         """
 | |
|         make slot data, which consists of yachtdice_data, options, and some other variables.
 | |
|         """
 | |
|         yacht_dice_data = self._get_yachtdice_data()
 | |
|         yacht_dice_options = self.options.as_dict(
 | |
|             "game_difficulty",
 | |
|             "score_for_last_check",
 | |
|             "score_for_goal",
 | |
|             "number_of_dice_fragments_per_dice",
 | |
|             "number_of_roll_fragments_per_roll",
 | |
|             "which_story",
 | |
|             "allow_manual_input",
 | |
|         )
 | |
|         slot_data = {**yacht_dice_data, **yacht_dice_options}  # combine the two
 | |
|         slot_data["number_of_dice_fragments_per_dice"] = self.frags_per_dice
 | |
|         slot_data["number_of_roll_fragments_per_roll"] = self.frags_per_roll
 | |
|         slot_data["goal_score"] = self.goal_score
 | |
|         slot_data["last_check_score"] = self.max_score
 | |
|         slot_data["allowed_categories"] = self.possible_categories
 | |
|         slot_data["ap_world_version"] = self.ap_world_version
 | |
|         return slot_data
 | |
| 
 | |
|     def create_item(self, name: str) -> Item:
 | |
|         item_data = item_table[name]
 | |
|         item = YachtDiceItem(name, item_data.classification, item_data.code, self.player)
 | |
|         return item
 | |
| 
 | |
|     # We overwrite these function to monitor when states have changed. See also dice_simulation in Rules.py
 | |
|     def collect(self, state: CollectionState, item: Item) -> bool:
 | |
|         change = super().collect(state, item)
 | |
|         if change:
 | |
|             state.prog_items[self.player]["state_is_fresh"] = 0
 | |
| 
 | |
|         return change
 | |
| 
 | |
|     def remove(self, state: CollectionState, item: Item) -> bool:
 | |
|         change = super().remove(state, item)
 | |
|         if change:
 | |
|             state.prog_items[self.player]["state_is_fresh"] = 0
 | |
| 
 | |
|         return change
 |