Files
Archipelago/worlds/stardew_valley/items/item_data.py
agilbert1412 1de91fab67 Stardew Valley: 7.x.x - The Jojapocalypse Update (#5432)
Major Content update for Stardew Valley

### Features
- New BundleRandomization Value: Meme Bundles - Over 100 custom bundles, designed to be jokes, references, trolls, etc
- New Setting: Bundles Per Room modifier
- New Setting: Backpack Size
- New Setting: Secretsanity - Checks for triggering easter eggs and secrets
- New Setting: Moviesanity - Checks for watching movies and sharing snacks with Villagers
- New Setting: Eatsanity - Checks for eating items
- New Setting: Hatsanity - Checks for wearing Hats
- New Setting: Start Without - Allows you to select any combination of various "starting" items, that you will actually not start with. Notably, tools, backpack slots, Day5 unlocks, etc.
- New Setting: Allowed Filler Items - Allows you to customize the filler items you'll get
- New Setting: Endgame Locations - Checks for various expensive endgame tasks and purchases
- New Shipsanity value: Crops and Fish
- New Settings: Jojapocalypse and settings to customize it
- Bundle Plando: Replaced with BundleWhitelist and BundleBlacklist, for more customization freedom
- Added a couple of Host.yaml settings to help hosts allow or ban specific difficult settings that could cause problems if the people don't know what they are signing up for.

Plus a truckload of improvements on the mod side, not seen in this PR.

### Removed features
- Integration for Stardew Valley Expanded. It is simply disabled, the code is all still there, but I'm extremely tired of providing tech support for it, plus Stardew Valley 1.7 was announced and that will break it again, so I'm done. When a maintainer steps up, it can be re-enabled.
2026-02-15 18:02:21 +01:00

198 lines
6.6 KiB
Python

from __future__ import annotations
import csv
import enum
from dataclasses import dataclass, field
from functools import reduce
from typing import Protocol
from BaseClasses import ItemClassification, Item
from .. import data
from ..content.vanilla.ginger_island import ginger_island_content_pack
from ..logic.logic_event import all_events
ITEM_CODE_OFFSET = 717000
class StardewItemFactory(Protocol):
def __call__(self, item: str | ItemData, /, *, classification_pre_fill: ItemClassification = None,
classification_post_fill: ItemClassification = None) -> Item:
"""
:param item: The item to create. Can be the name of the item or the item data.
:param classification_pre_fill: The classification to use for the item before the fill. If None, the basic classification of the item is used.
:param classification_post_fill: The classification to use for the item after the fill. If None, the pre_fill classification will be used.
"""
raise NotImplementedError
class Group(enum.Enum):
FRIENDSHIP_PACK = enum.auto()
COMMUNITY_REWARD = enum.auto()
TRASH_BEAR = enum.auto()
TRASH = enum.auto()
FOOTWEAR = enum.auto()
WEAPON = enum.auto()
WEAPON_GENERIC = enum.auto()
WEAPON_SWORD = enum.auto()
WEAPON_CLUB = enum.auto()
WEAPON_DAGGER = enum.auto()
WEAPON_SLINGSHOT = enum.auto()
PROGRESSIVE_TOOLS = enum.auto()
SKILL_LEVEL_UP = enum.auto()
SKILL_MASTERY = enum.auto()
BUILDING = enum.auto()
WIZARD_BUILDING = enum.auto()
DESERT_TRANSPORTATION = enum.auto()
ISLAND_TRANSPORTATION = enum.auto()
ARCADE_MACHINE_BUFFS = enum.auto()
BASE_RESOURCE = enum.auto()
WARP_TOTEM = enum.auto()
GEODE = enum.auto()
ORE = enum.auto()
FERTILIZER = enum.auto()
CROPSANITY = enum.auto()
FISHING_RESOURCE = enum.auto()
SEASON = enum.auto()
TRAVELING_MERCHANT_DAY = enum.auto()
MUSEUM = enum.auto()
FRIENDSANITY = enum.auto()
FESTIVAL = enum.auto()
RARECROW = enum.auto()
TRAP = enum.auto()
BONUS = enum.auto()
MAXIMUM_ONE = enum.auto()
AT_LEAST_TWO = enum.auto()
DEPRECATED = enum.auto()
SPECIAL_ORDER_BOARD = enum.auto()
SPECIAL_ORDER_QI = enum.auto()
BABY = enum.auto()
GINGER_ISLAND = enum.auto()
WALNUT_PURCHASE = enum.auto()
TV_CHANNEL = enum.auto()
QI_CRAFTING_RECIPE = enum.auto()
CHEFSANITY = enum.auto()
CHEFSANITY_STARTER = enum.auto()
CHEFSANITY_QOS = enum.auto()
CHEFSANITY_PURCHASE = enum.auto()
CHEFSANITY_FRIENDSHIP = enum.auto()
CHEFSANITY_SKILL = enum.auto()
CRAFTSANITY = enum.auto()
BOOK_POWER = enum.auto()
LOST_BOOK = enum.auto()
PLAYER_BUFF = enum.auto()
EASY_SECRET = enum.auto()
FISHING_SECRET = enum.auto()
SECRET_NOTES_SECRET = enum.auto()
MOVIESANITY = enum.auto()
TRINKET = enum.auto()
EATSANITY_ENZYME = enum.auto()
ENDGAME_LOCATION_ITEMS = enum.auto()
REQUIRES_FRIENDSANITY_MARRIAGE = enum.auto()
BOOKSELLER = enum.auto()
# Types of filler
FILLER_FARMING = enum.auto()
FILLER_FISHING = enum.auto()
FILLER_FRUIT_TREES = enum.auto()
FILLER_FOOD = enum.auto()
FILLER_BUFF_FOOD = enum.auto()
FILLER_CONSUMABLE = enum.auto()
FILLER_MACHINE = enum.auto()
FILLER_STORAGE = enum.auto()
FILLER_QUALITY_OF_LIFE = enum.auto()
FILLER_MATERIALS = enum.auto()
FILLER_CURRENCY = enum.auto()
FILLER_MONEY = enum.auto()
FILLER_HAT = enum.auto()
FILLER_DECORATION = enum.auto()
FILLER_RING = enum.auto()
# Mods
MAGIC_SPELL = enum.auto()
MOD_WARP = enum.auto()
FILLER_GROUPS = [Group.FILLER_FARMING, Group.FILLER_FISHING, Group.FILLER_FRUIT_TREES, Group.FILLER_FOOD, Group.FILLER_BUFF_FOOD,
Group.FILLER_CONSUMABLE, Group.FILLER_MACHINE, Group.FILLER_STORAGE, Group.FILLER_QUALITY_OF_LIFE, Group.FILLER_MATERIALS,
Group.FILLER_CURRENCY, Group.FILLER_MONEY, Group.FILLER_HAT, Group.FILLER_DECORATION, Group.FILLER_RING, ]
@dataclass(frozen=True)
class ItemData:
code_without_offset: int | None
name: str
classification: ItemClassification
content_packs: frozenset[str] = frozenset()
"""All the content packs required for this item to be available."""
groups: set[Group] = field(default_factory=frozenset)
def __post_init__(self):
if not isinstance(self.groups, frozenset):
super().__setattr__("groups", frozenset(self.groups))
@property
def code(self) -> int | None:
return ITEM_CODE_OFFSET + self.code_without_offset if self.code_without_offset is not None else None
def has_any_group(self, *group: Group) -> bool:
groups = set(group)
return bool(groups.intersection(self.groups))
def has_all_groups(self, *group: Group) -> bool:
groups = set(group)
return bool(groups.issubset(self.groups))
def has_limited_amount(self) -> bool:
return self.has_any_group(Group.MAXIMUM_ONE, Group.AT_LEAST_TWO)
def load_item_csv():
from importlib.resources import files
items = []
with files(data).joinpath("items.csv").open() as file:
item_reader = csv.DictReader(file)
for item in item_reader:
item_id = int(item["id"]) if item["id"] else None
item_name = item["name"]
classification = reduce((lambda a, b: a | b), {ItemClassification[str_classification] for str_classification in item["classification"].split(",")})
csv_groups = [Group[group] for group in item["groups"].split(",") if group]
groups = set(csv_groups)
csv_content_packs = [cp for cp in item["content_packs"].split(",") if cp]
content_packs = frozenset(csv_content_packs)
assert len(csv_groups) == len(groups), f"Item '{item_name}' has duplicate groups: {csv_groups}"
assert len(csv_content_packs) == len(content_packs)
if Group.GINGER_ISLAND in groups:
content_packs |= {ginger_island_content_pack.name}
items.append(ItemData(item_id, item_name, classification, content_packs, groups))
return items
events = [
ItemData(None, e, ItemClassification.progression)
for e in sorted(all_events)
]
all_items: list[ItemData] = load_item_csv() + events
item_table: dict[str, ItemData] = {}
items_by_group: dict[Group, list[ItemData]] = {}
def initialize_groups():
for item in all_items:
for group in item.groups:
item_group = items_by_group.get(group, list())
item_group.append(item)
items_by_group[group] = item_group
def initialize_item_table():
item_table.update({item.name: item for item in all_items})
initialize_item_table()
initialize_groups()