rimlike/autoload/strings.gd
megaproxy 19d28ca9f8 Phase 16: Save/load full coverage + autosave + UI
Three-agent fan-out reusing the contracts-first pattern: Opus pre-wrote
World.clear_all + 4 EventBus signals (save_started/finished, load_started/
finished) before dispatch. Pattern proven across Phases 12/13/14/15/16.

Entity to_dict/from_dict + class_id tagging (Agent A):
- class_id tag added to all 18 entity to_dict methods for loader routing
- Missing pairs filled in: wolf, grave_slot, graveyard_zone, stockpile_zone,
  crate (from_dict). All defensive with d.get(field, default).
- Workbench round-trips label_text so Carpenter/Smelter/Millstone/Hearth/
  Pyre kinds survive reload
- BeautySystem + DirtinessSystem save_dict/apply_dict for sparse maps
- World.save_tilemap_layers / apply_tilemap_layers covering 5 layers
  (Terrain/Floor/Wall/Designation/Roof; Fog runtime-only skipped)

SaveSystem v2 rewrite (Agent B):
- SAVE_VERSION bumped from 1 to 2
- write_save(slot) pauses Sim, emits save_started, collects every entity
  via _collect_entities iterating all World registries, writes payload to
  user://save_<slot>.json
- apply_save full rewrite: pause sim → emit load_started → World.clear_all
  → apply autoloads (GameState/Clock/Weather/Storyteller) → apply tilemap
  layers → iterate payload.entities and dispatch to per-class factories
  → apply beauty/dirt maps → emit load_finished(slot, ok, real_seconds_away)
- Per-class factory registry: 18 class_ids dispatched to setup+add_child+
  from_dict patterns. CremationPyre detected via workbench.label_text == 'Pyre'
- Public slot API: save_to_slot/load_from_slot/has_save/delete_save/
  peek_save_metadata. Slots locked: &manual + &autosave

Autosave + UI + Resume toast (Agent C):
- autoload/autosave.gd — new Autosave autoload. Periodic every
  AUTOSAVE_INTERVAL_TICKS = 6000 (~5 in-game min at 20 Hz) + NOTIFICATION_
  APPLICATION_PAUSED (mobile) + NOTIFICATION_WM_WINDOW_FOCUS_OUT (desktop).
  Gated by _busy flag tied to EventBus.save_started/save_finished.
- TopBar extended with SaveBtn (💾) + LoadBtn buttons, 48×48 min hit area
- scenes/ui/load_menu.gd — CanvasLayer slot picker. Reads peek_save_metadata
  to show 'Manual save (Date Time)' / 'Autosave (Date Time)' rows.
  Version-mismatch warning dialog before continuing on older saves.
- scenes/ui/resume_toast.gd — top-center toast. On load_finished(ok=true):
  'Welcome back — N minutes/hours away' for 5s + 0.8s fade.
  On ok=false: 'Load failed (corrupt or version mismatch)'.
- Strings catalog: 14 new keys (ui.save / ui.load / ui.welcome_back_* /
  ui.load_failed etc.)
- main.gd mounts LoadMenu + ResumeToast as runtime CanvasLayer children

MCP runtime verified:
- Saved at tick 1137 → [save] wrote slot 'manual': 113 entities at tick 1137
- Advanced sim to tick 4600 at ULTRA speed (different state)
- load_from_slot(&manual) → [save] applied slot 'manual': 113 entities,
  0 errors, tick=1137, away=34s
- post-load: Sim.tick=1137 (restored), pawns alive=3, all furniture +
  workbenches + crops + walls + floors back in place
- Resume toast fires: [resume_toast] showing — ok=true seconds_away=34
- Autosave on focus-loss verified: [autosave] focus-loss → wrote autosave
- Screenshot shows TopBar with Save + Load buttons + post-load Lone Wolf
  storyteller modal from fresh dawn roll

Known acceptable gaps (deferred to Phase 20 tuning):
- Pawn JobRunner mid-INTERACT/mid-BUILD restarts from toil 0 on reload
  (walk toil round-trips; multi-step interact does not). Pawns lose a few
  seconds of work.
- Workbench bill mid-craft fetch state isn't fully serialized.
- Wolf.target_pawn re-resolution from name string is Agent A's documented
  pattern; Agent B's apply_save respects pawn-restoration ordering so the
  resolution works after pawns are back.

Delegation: 3× gdscript-refactor (Sonnet) agents in parallel; integration
+ MCP verify on Opus.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-11 19:24:59 +01:00

142 lines
7.5 KiB
GDScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

extends Node
## i18n string table — player-visible strings ONLY. Code keys, EN values.
##
## Locked from day one (per CLAUDE.md): no hardcoded display copy in scenes or
## scripts. If you have player-facing text, add a key here and call Strings.t(key).
##
## Locale switching is post-MVP; the indirection lands now so we don't have to
## retrofit the whole game later. When the table grows, move it to a .tres or
## external CSV import; the public API (`Strings.t(key)`) stays the same.
const TABLE: Dictionary = {
# Phase 0 placeholder — populate as features land.
&"app.title": "Rimlike",
&"smoke.hello": "Phase 0 — autoloads online.",
# Speed controls (top bar)
&"speed.pause": "",
&"speed.normal": "1×",
&"speed.fast": "5×",
&"speed.ultra": "12×",
# HUD
&"hud.tick": "Tick: {n}",
# Phase 11 — in-game clock display ("{d}" = day, "{t}" = "HH:MM")
&"clock.format": "Day {d}, {t}",
# Phase 12 — season indicator ("{s}" = season name, "{d}" = 1-indexed day, "{total}" = days per season)
&"season.spring": "Spring",
&"season.summer": "Summer",
&"season.autumn": "Autumn",
&"season.winter": "Winter",
&"season.format": "{s} {d}/12",
# Pawn state labels
&"pawn.state.idle": "idle",
&"pawn.state.walking": "walking",
# Item types (player-visible this phase)
&"item.wood": "Wood",
&"item.stone": "Stone",
&"item.iron_ore": "Iron ore",
# Item stack count badge ("{n}" is substituted at call site via .format())
&"item.stack_count": "×{n}",
# Phase 6 — new item types (carpenter bench + smelter outputs)
&"item.plank": "Plank",
&"item.stone_block": "Stone block",
# Phase 7 — food loop and cooking chain item types
&"item.flour": "Flour",
&"item.bread": "Bread",
&"item.meal": "Meal",
# Phase 7 — cooking workbench labels
&"workbench.hearth": "Hearth",
&"workbench.millstone": "Millstone",
# Phase 7 — pawn hunger states
&"pawn.state.eating": "eating",
&"pawn.state.hungry": "hungry",
# Phase 11 — mood thoughts (player-visible in pawn-detail, Phase 17)
&"thought.in_darkness": "In darkness",
# Phase 6 — quality tier labels
&"quality.shoddy": "Shoddy",
&"quality.normal": "Normal",
&"quality.excellent": "Excellent",
&"quality.masterwork": "Masterwork",
&"quality.legendary": "Legendary",
# Phase 15 — Storyteller UI buttons
&"ui.go_there": "Go there",
&"ui.dismiss": "Dismiss",
# Phase 16 — Save/Load UI
&"ui.save": "Save",
&"ui.load": "Load",
&"ui.saved": "Saved",
&"ui.saving": "Saving…",
&"ui.no_saves": "No saves yet.",
&"ui.continue": "Continue",
&"ui.cancel": "Cancel",
&"ui.manual_save": "Manual save",
&"ui.autosave": "Autosave",
&"ui.version_mismatch": "This save is from an older version (v{v}) — loading may fail. Continue?",
&"ui.welcome_back": "Welcome back — away {n}",
&"ui.welcome_back_min": "{n} minute",
&"ui.welcome_back_mins": "{n} minutes",
&"ui.welcome_back_hour": "{n} hour",
&"ui.welcome_back_hours": "{n} hours",
&"ui.load_failed": "Load failed (corrupt or version mismatch).",
# Phase 15 — Storyteller event titles + bodies (25-event corpus).
# EventDef factories in EventCatalog carry the English string directly;
# these keys exist so Strings.t() is the indirection point for future locale
# switching. When a locale ships, swap EventDef.title/body from these keys
# instead of touching EventCatalog factories. (%pawn% is substituted by UI.)
&"event.first_beds.title": "First Beds",
&"event.first_beds.body": "Your settlers slept on the cold ground again. They are starting to ache.",
&"event.empty_larder.title": "Empty Larder",
&"event.empty_larder.body": "The larder is bare. Spring won't last forever.",
&"event.no_fire.title": "No Fire",
&"event.no_fire.body": "Without a hearth, the cold will bite by night.",
&"event.walls.title": "Walls?",
&"event.walls.body": "Sleeping under stars is romantic until the wolves arrive.",
&"event.spring_awakens.title": "Spring Awakens",
&"event.spring_awakens.body": "The thaw runs in every stream. Crops will grow fast now.",
&"event.summers_heat.title": "Summer's Heat",
&"event.summers_heat.body": "The sun beats down. Unsheltered work will tire faster.",
&"event.autumns_harvest.title": "Autumn's Harvest",
&"event.autumns_harvest.body": "The fields are heavy with the last of the year's bounty.",
&"event.winters_edge.title": "Winter's Edge",
&"event.winters_edge.body": "Frost has come. The road is closed; you are alone.",
&"event.a_traveler.title": "A Traveler",
&"event.a_traveler.body": "A weary traveler stumbles toward your gate. They look hungry. Will you welcome them?",
&"event.the_refugee_family.title": "The Refugee Family",
&"event.the_refugee_family.body": "A family fleeing bandits arrives. They have nothing, but they would work hard.",
&"event.the_old_soldier.title": "The Old Soldier",
&"event.the_old_soldier.body": "A retired soldier offers his blade for a place by your fire. Combat 8, but old and tired.",
&"event.the_wandering_healer.title": "The Wandering Healer",
&"event.the_wandering_healer.body": "A traveling healer asks for shelter. She brings knowledge of medicine.",
&"event.wolves_at_the_edge.title": "Wolves at the Edge",
&"event.wolves_at_the_edge.body": "Wolves howl in the distance. They will be here by nightfall.",
&"event.lone_wolf.title": "Lone Wolf",
&"event.lone_wolf.body": "A starving wolf circles your livestock.",
&"event.pack_hunt.title": "Pack Hunt",
&"event.pack_hunt.body": "A hunting pack moves through the forest. They smell your colony.",
&"event.bandit_scouts.title": "Bandit Scouts",
&"event.bandit_scouts.body": "Strange figures watched from the treeline at dusk. Bandits, perhaps.",
&"event.fever.title": "Fever",
&"event.fever.body": "%pawn% woke with a fever. The sickness may spread.",
&"event.a_bad_cut.title": "A Bad Cut",
&"event.a_bad_cut.body": "%pawn% gashed their hand chopping wood. The wound looks deep.",
&"event.the_sleeplessness.title": "The Sleeplessness",
&"event.the_sleeplessness.body": "%pawn% has barely slept. Something weighs on them.",
&"event.bountiful_harvest.title": "Bountiful Harvest",
&"event.bountiful_harvest.body": "Your fields exceeded the season. The granary swells.",
&"event.lumberjacks_luck.title": "Lumberjack's Luck",
&"event.lumberjacks_luck.body": "%pawn% found a copse of unusually thick trees.",
&"event.veins_of_iron.title": "Veins of Iron",
&"event.veins_of_iron.body": "A miner reports a rich vein, deeper than expected.",
&"event.strange_stones.title": "Strange Stones",
&"event.strange_stones.body": "Settlers report finding carved stones in the wood — older than any memory.",
&"event.an_old_map.title": "An Old Map",
&"event.an_old_map.body": "%pawn% found a tattered map. Roads to the north, half-faded.",
&"event.one_year_survived.title": "One Year Survived",
&"event.one_year_survived.body": "A full year. The first frost feels different now — yours is a real settlement.",
}
func t(key: StringName) -> String:
if TABLE.has(key):
return TABLE[key]
push_warning("Strings.t(): missing key %s" % key)
return String(key)