class_name CraftingProvider extends WorkProvider ## WorkProvider for the Crafting work category. Slots into the 5-layer pawn AI ## (Decision → WorkProvider → Job + JobRunner) as layer 2. ## ## Each call to find_best_for(pawn) scans World.workbenches for the best ## (Workbench, Bill) pair the pawn qualifies for, then builds a 4-toil job: ## walk_to(ingredient.tile) → pickup → walk_to(wb.tile) → craft_at(wb, bill_index) ## ## Scoring: Manhattan distance pawn→ingredient + ingredient→workbench (lower wins). ## ## Phase 6 simplification: pawn must be empty-handed (one task at a time); ingredient ## search is global (no per-bench radius restriction — Phase 17 polish item per ## docs/architecture.md "Ingredient acquisition radius"). ## ## When a bill cannot proceed it emits EventBus.bill_blocked once per ## (workbench × reason) per BILL_BLOCKED_COOLDOWN_TICKS (60 in-game seconds). ## ## Workbench and Pawn are intentionally duck-typed (no class_name reference) to match ## WorkProvider convention and avoid init-order issues. Only Item.Quality and ## QualityCalc are referenced by class_name (in job_runner.gd, not here). ## ## See docs/architecture.md "CraftingProvider" and docs/design.md "Bills". ## Rate-limit for bill_blocked alerts: one emit per (workbench × reason) per ## 60 in-game seconds (20 Hz × 60 s = 1200 ticks). const BILL_BLOCKED_COOLDOWN_TICKS: int = 1200 ## Per-(workbench_id|reason) cooldown map: String → tick at which next emit is allowed. var _bill_blocked_cooldown: Dictionary = {} func _init() -> void: category = &"crafting" # Priority 4 — above haul (3), below chop (5). # Phase 6 demo ordering; final 9-category matrix is authored in Phase 17. priority = 4 # ── WorkProvider override ───────────────────────────────────────────────────── ## Returns a craft Job for `pawn`, or null if no valid work exists. ## Pawn must expose: .carried_item, .tile (Vector2i), .get_skill(StringName) -> int. func find_best_for(pawn) -> Job: # Skip if pawn is already carrying something — deposit first. if pawn.get("carried_item") != null: return null var best_wb = null var best_bill = null var best_bill_index: int = -1 var best_src = null var best_dist: int = 999999 for wb in World.workbenches: # Duck-type guard: skip workbenches that aren't fully set up yet. if not wb.get("_completed"): continue # Skip workbenches another pawn has already claimed for crafting. if Job.is_target_taken_by_other(wb, pawn): continue for i in wb.bills.size(): var b = wb.bills[i] if not b.is_active(): continue # Skill threshold check — pawn must meet the bill's minimum. if pawn.get_skill(b.recipe.required_skill) < b.recipe.skill_threshold: _emit_bill_blocked(b.recipe.label, &"skill_too_low", wb) continue # If ingredient_count is 0, no ingredient is required; proceed directly. # Otherwise, confirm a qualifying ingredient exists on the floor. var src = null if b.recipe.ingredient_count > 0: src = _find_ingredient_item(b.recipe.ingredient_type) if src == null: _emit_bill_blocked(b.recipe.label, &"missing_ingredient", wb) continue # Score: total Manhattan travel distance. # If no ingredient (count==0), distance is just pawn → workbench. # Otherwise, distance is pawn → ingredient → workbench. var d: int if b.recipe.ingredient_count > 0: d = _manhattan(pawn.tile, src.tile) + _manhattan(src.tile, wb.tile) else: d = _manhattan(pawn.tile, wb.tile) if d < best_dist: best_dist = d best_wb = wb best_bill = b best_bill_index = i best_src = src if best_wb == null: return null var src_item = null # If ingredient_count > 0, re-resolve the source item in case multiple bills tied on the same item. if best_bill.recipe.ingredient_count > 0: src_item = _find_ingredient_item(best_bill.recipe.ingredient_type) if src_item == null: return null var j := Job.new() j.label = "Craft %s at %s" % [best_bill.recipe.label, best_wb.get("label_text") if best_wb.get("label_text") != null else "workbench"] j.target_node = best_wb # Only add ingredient-haul toils if ingredient is required. if best_bill.recipe.ingredient_count > 0: j.toils.append(Toil.walk_to(src_item.tile)) j.toils.append(Toil.pickup()) j.toils.append(Toil.walk_to(best_wb.tile)) j.toils.append(Toil.craft_at(best_wb.get_path(), best_bill_index)) return j # ── private helpers ─────────────────────────────────────────────────────────── ## Returns the first on-floor Item of matching type that is not being carried. ## Phase 6 simplification: global search, first match wins (no nearest-first ## at this layer — distance is factored into the outer loop scoring instead). func _find_ingredient_item(item_type: StringName): for it in World.items: if it.being_carried: continue if it.item_type == item_type: return it return null ## Manhattan distance between two Vector2i tile coordinates. func _manhattan(a: Vector2i, b: Vector2i) -> int: return abs(a.x - b.x) + abs(a.y - b.y) ## Emits EventBus.bill_blocked for the given workbench and reason, rate-limited ## to once per (workbench × reason) per BILL_BLOCKED_COOLDOWN_TICKS. func _emit_bill_blocked(recipe_label: String, reason: StringName, wb) -> void: var key: String = "%s|%s" % [wb.get_instance_id(), reason] if _bill_blocked_cooldown.get(key, 0) > Sim.tick: return _bill_blocked_cooldown[key] = Sim.tick + BILL_BLOCKED_COOLDOWN_TICKS var focus: Vector2i = wb.get("tile") if wb.get("tile") != null else Vector2i(-1, -1) EventBus.bill_blocked.emit(recipe_label, reason, focus)