* fix vanilla tool fishing rod requiring metal bars fix vanilla skill requiring previous level (it's always the same rule or more restrictive) * add test to ensure fishing rod need fish shop * fishing rod should be indexed from 0 like a mentally sane person would do. * fishing rod 0 isn't real, but it definitely can hurt you. * reeeeeeeee
		
			
				
	
	
		
			90 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			90 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
from typing import Union, Iterable
 | 
						|
 | 
						|
from Utils import cache_self1
 | 
						|
from .base_logic import BaseLogicMixin, BaseLogic
 | 
						|
from .has_logic import HasLogicMixin
 | 
						|
from .money_logic import MoneyLogicMixin
 | 
						|
from .received_logic import ReceivedLogicMixin
 | 
						|
from .region_logic import RegionLogicMixin
 | 
						|
from .season_logic import SeasonLogicMixin
 | 
						|
from ..mods.logic.magic_logic import MagicLogicMixin
 | 
						|
from ..options import ToolProgression
 | 
						|
from ..stardew_rule import StardewRule, True_, False_
 | 
						|
from ..strings.ap_names.skill_level_names import ModSkillLevel
 | 
						|
from ..strings.region_names import Region
 | 
						|
from ..strings.spells import MagicSpell
 | 
						|
from ..strings.tool_names import ToolMaterial, Tool
 | 
						|
 | 
						|
fishing_rod_prices = {
 | 
						|
    3: 1800,
 | 
						|
    4: 7500,
 | 
						|
}
 | 
						|
 | 
						|
tool_materials = {
 | 
						|
    ToolMaterial.copper: 1,
 | 
						|
    ToolMaterial.iron: 2,
 | 
						|
    ToolMaterial.gold: 3,
 | 
						|
    ToolMaterial.iridium: 4
 | 
						|
}
 | 
						|
 | 
						|
tool_upgrade_prices = {
 | 
						|
    ToolMaterial.copper: 2000,
 | 
						|
    ToolMaterial.iron: 5000,
 | 
						|
    ToolMaterial.gold: 10000,
 | 
						|
    ToolMaterial.iridium: 25000
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
class ToolLogicMixin(BaseLogicMixin):
 | 
						|
    def __init__(self, *args, **kwargs):
 | 
						|
        super().__init__(*args, **kwargs)
 | 
						|
        self.tool = ToolLogic(*args, **kwargs)
 | 
						|
 | 
						|
 | 
						|
class ToolLogic(BaseLogic[Union[ToolLogicMixin, HasLogicMixin, ReceivedLogicMixin, RegionLogicMixin, SeasonLogicMixin, MoneyLogicMixin, MagicLogicMixin]]):
 | 
						|
    # Should be cached
 | 
						|
    def has_tool(self, tool: str, material: str = ToolMaterial.basic) -> StardewRule:
 | 
						|
        assert tool != Tool.fishing_rod, "Use `has_fishing_rod` instead of `has_tool`."
 | 
						|
 | 
						|
        if material == ToolMaterial.basic or tool == Tool.scythe:
 | 
						|
            return True_()
 | 
						|
 | 
						|
        if self.options.tool_progression & ToolProgression.option_progressive:
 | 
						|
            return self.logic.received(f"Progressive {tool}", tool_materials[material])
 | 
						|
 | 
						|
        return self.logic.has(f"{material} Bar") & self.logic.money.can_spend_at(Region.blacksmith, tool_upgrade_prices[material])
 | 
						|
 | 
						|
    def can_use_tool_at(self, tool: str, material: str, region: str) -> StardewRule:
 | 
						|
        return self.has_tool(tool, material) & self.logic.region.can_reach(region)
 | 
						|
 | 
						|
    @cache_self1
 | 
						|
    def has_fishing_rod(self, level: int) -> StardewRule:
 | 
						|
        assert 1 <= level <= 4, "Fishing rod 0 isn't real, it can't hurt you. Training is 1, Bamboo is 2, Fiberglass is 3 and Iridium is 4."
 | 
						|
 | 
						|
        if self.options.tool_progression & ToolProgression.option_progressive:
 | 
						|
            return self.logic.received(f"Progressive {Tool.fishing_rod}", level)
 | 
						|
 | 
						|
        if level <= 2:
 | 
						|
            # We assume you always have access to the Bamboo pole, because mod side there is a builtin way to get it back.
 | 
						|
            return self.logic.region.can_reach(Region.beach)
 | 
						|
 | 
						|
        return self.logic.money.can_spend_at(Region.fish_shop, fishing_rod_prices[level])
 | 
						|
 | 
						|
    # Should be cached
 | 
						|
    def can_forage(self, season: Union[str, Iterable[str]], region: str = Region.forest, need_hoe: bool = False) -> StardewRule:
 | 
						|
        season_rule = False_()
 | 
						|
        if isinstance(season, str):
 | 
						|
            season_rule = self.logic.season.has(season)
 | 
						|
        elif isinstance(season, Iterable):
 | 
						|
            season_rule = self.logic.season.has_any(season)
 | 
						|
        region_rule = self.logic.region.can_reach(region)
 | 
						|
        if need_hoe:
 | 
						|
            return season_rule & region_rule & self.logic.tool.has_tool(Tool.hoe)
 | 
						|
        return season_rule & region_rule
 | 
						|
 | 
						|
    @cache_self1
 | 
						|
    def can_water(self, level: int) -> StardewRule:
 | 
						|
        tool_rule = self.logic.tool.has_tool(Tool.watering_can, ToolMaterial.tiers[level])
 | 
						|
        spell_rule = self.logic.received(MagicSpell.water) & self.logic.magic.can_use_altar() & self.logic.received(ModSkillLevel.magic_level, level)
 | 
						|
        return tool_rule | spell_rule
 |