rimlike/autoload/audio.gd
megaproxy d819c13a9d Phase 18 — Audio (music director + SFX catalog + bus wiring)
Adds an AudioManager autoload with three buses (Master, Music routed to
Master, SFX routed to Master), a small catalog of looping music + one-shot
SFX, and a single persistent AudioStreamPlayer for the music director.

Music
* Day and night loops swap on Clock.phase_changed (night during the night
  phase, day everywhere else). Tracks pulled from Retro Farming Music 1
  (day) and Cozy Melodies Pack 1 (night), both loopable OGG.

SFX
* Tree.fell, Rock.mined, BigRock.mined → tree_fell / mine_tick.
* EventBus.pawn_took_damage → combat_hit (Sword Pack 1).
* EventBus.storyteller_event_fired → ui_confirm sting.
* EventBus.alert_added → ui_click.
* play_sfx is rate-limited per key (80ms cooldown) so fast-sim doesn't
  saturate the mixer.

Settings + suspend
* SettingsMenu master/music/sfx sliders now live-bind to the bus dB via
  Audio.set_*_linear (linear → dB internally, 0 → -80dB silence). The
  ambient slider is intentionally unwired; no ambient bus this pass.
* NOTIFICATION_APPLICATION_PAUSED + FOCUS_OUT mute the Master bus to
  match the existing "no background sim" rule. Resume + focus restore it.

Bundle housekeeping
* Two zipped packs in the ElvGames bundle (Cozy Melodies Pack 1, Retro
  Farming Music 1) extracted in place to keep pack identity intact for
  the license/credits string. 8 OGG files curated into audio/ at ~5.3MB.

Verified end-to-end via MCP runtime: buses online, day_loop plays at
boot, manual phase swap day→night→day round-trips, slider linear→dB
mapping correct (0.5 → -6.02dB, 0.0 → -80dB), tree_fell SFX triggers.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-15 18:54:36 +01:00

234 lines
8.2 KiB
GDScript

extends Node
## AudioManager — Phase 18.
##
## Owns three AudioServer buses (Master, Music, SFX), a small catalog of music
## loops + one-shot SFX, and a single AudioStreamPlayer for the music director.
## SFX play through pooled short-lived AudioStreamPlayers.
##
## Music swaps on Clock.phase_changed (day/dusk/night/dawn). SFX fire on
## EventBus signals (combat, storyteller stings) and direct calls from Tree /
## Rock entities for chop / mine completion sounds.
##
## Volume sliders in SettingsMenu drive set_*_db() at the Master/Music/SFX bus
## level. App-pause mutes everything (NOTIFICATION_APPLICATION_PAUSED / focus
## loss) to match the existing "no background simulation" rule.
const BUS_MASTER: StringName = &"Master"
const BUS_MUSIC: StringName = &"Music"
const BUS_SFX: StringName = &"SFX"
# Slider range is 0.0..1.0 (linear); we map to dB on the bus.
const SILENT_DB: float = -80.0
# How long the chop/mine SFX cooldown is to avoid spam when many entities work
# simultaneously. Measured in milliseconds.
const SFX_THROTTLE_MS: int = 80
## Catalog: StringName → AudioStream. Loaded lazily on first access; once a
## key resolves we cache the stream so the second play is a Dictionary lookup.
const SFX_FILES: Dictionary = {
&"ui_click": "res://audio/sfx/ui_click.ogg",
&"ui_confirm": "res://audio/sfx/ui_confirm.ogg",
&"chop_tick": "res://audio/sfx/chop_tick.ogg",
&"tree_fell": "res://audio/sfx/tree_fell.ogg",
&"mine_tick": "res://audio/sfx/mine_tick.ogg",
&"combat_hit": "res://audio/sfx/combat_hit.ogg",
}
const MUSIC_FILES: Dictionary = {
&"day_loop": "res://audio/music/day_loop.ogg",
&"night_loop": "res://audio/music/night_loop.ogg",
}
# Loaded streams. Populated on demand.
var _sfx_streams: Dictionary = {}
var _music_streams: Dictionary = {}
# Single persistent music player. Stream is swapped on phase change.
var _music_player: AudioStreamPlayer = null
var _current_music_key: StringName = &""
# Per-SFX cooldowns to avoid spam. Last-played epoch ms keyed by SFX key.
var _sfx_last_played_ms: Dictionary = {}
# Cached slider values (linear 0..1). Applied to buses on _ready and on
# settings change. Public so SettingsMenu can rebind after creation.
var master_linear: float = 1.0
var music_linear: float = 1.0
var sfx_linear: float = 1.0
# ── lifecycle ───────────────────────────────────────────────────────────────
func _ready() -> void:
_ensure_buses()
# Music director: track stays on a single persistent player; stream swaps.
_music_player = AudioStreamPlayer.new()
_music_player.bus = BUS_MUSIC
_music_player.name = "MusicPlayer"
add_child(_music_player)
# SFX → world event wiring. Keep this list short and obvious.
if EventBus.has_signal("pawn_took_damage"):
EventBus.pawn_took_damage.connect(_on_pawn_took_damage)
if EventBus.has_signal("storyteller_event_fired"):
EventBus.storyteller_event_fired.connect(_on_storyteller_event_fired)
if EventBus.has_signal("alert_added"):
EventBus.alert_added.connect(_on_alert_added)
# Music director: swap day/night loops as Clock.phase_changed fires.
if Clock.has_signal("phase_changed"):
Clock.phase_changed.connect(_on_clock_phase_changed)
# Kick off the right initial track based on current phase, with no fade.
var initial_phase: StringName = &"day"
if Clock.has_method("current_phase"):
initial_phase = Clock.current_phase()
_apply_phase_to_music(initial_phase)
Audit.log("audio", "AudioManager ready (master/music/sfx buses online)")
## NOTIFICATION_APPLICATION_PAUSED fires when the app is suspended (mobile
## background, Steam Deck sleep). Mute Master so any in-flight SFX go silent;
## resume on focus.
func _notification(what: int) -> void:
match what:
NOTIFICATION_APPLICATION_PAUSED, NOTIFICATION_APPLICATION_FOCUS_OUT:
AudioServer.set_bus_mute(_bus_idx(BUS_MASTER), true)
NOTIFICATION_APPLICATION_RESUMED, NOTIFICATION_APPLICATION_FOCUS_IN:
AudioServer.set_bus_mute(_bus_idx(BUS_MASTER), false)
# ── public API ──────────────────────────────────────────────────────────────
## Play a one-shot SFX by catalog key. Quietly no-ops on unknown keys.
## Rate-limits by key so chop/mine tick spam doesn't clip the audio mixer.
func play_sfx(key: StringName) -> void:
if not SFX_FILES.has(key):
return
var now: int = Time.get_ticks_msec()
var last: int = int(_sfx_last_played_ms.get(key, -SFX_THROTTLE_MS))
if now - last < SFX_THROTTLE_MS:
return
_sfx_last_played_ms[key] = now
var stream: AudioStream = _load_sfx(key)
if stream == null:
return
var p := AudioStreamPlayer.new()
p.bus = BUS_SFX
p.stream = stream
add_child(p)
p.finished.connect(p.queue_free)
p.play()
## Switch the music track. `key` resolves through MUSIC_FILES. Same key as
## currently playing is a no-op so phase ticks don't restart the loop.
func play_music(key: StringName) -> void:
if key == _current_music_key:
return
if not MUSIC_FILES.has(key):
return
var stream: AudioStream = _load_music(key)
if stream == null:
return
_current_music_key = key
_music_player.stream = stream
_music_player.play()
## Slider → bus volume helpers. Linear 0..1; 0 mutes the bus.
func set_master_linear(linear: float) -> void:
master_linear = clampf(linear, 0.0, 1.0)
_apply_bus_db(BUS_MASTER, master_linear)
func set_music_linear(linear: float) -> void:
music_linear = clampf(linear, 0.0, 1.0)
_apply_bus_db(BUS_MUSIC, music_linear)
func set_sfx_linear(linear: float) -> void:
sfx_linear = clampf(linear, 0.0, 1.0)
_apply_bus_db(BUS_SFX, sfx_linear)
# ── internal ────────────────────────────────────────────────────────────────
func _ensure_buses() -> void:
# Godot ships with a single "Master" bus. We add Music + SFX as children
# routed to Master, so the master slider scales everything together.
_add_bus_if_missing(BUS_MUSIC, BUS_MASTER)
_add_bus_if_missing(BUS_SFX, BUS_MASTER)
func _add_bus_if_missing(bus_name: StringName, send_to: StringName) -> void:
if AudioServer.get_bus_index(bus_name) != -1:
return
var idx: int = AudioServer.bus_count
AudioServer.add_bus(idx)
AudioServer.set_bus_name(idx, bus_name)
AudioServer.set_bus_send(idx, send_to)
func _bus_idx(bus_name: StringName) -> int:
return AudioServer.get_bus_index(bus_name)
func _apply_bus_db(bus_name: StringName, linear: float) -> void:
var idx: int = _bus_idx(bus_name)
if idx < 0:
return
var db: float = linear_to_db(linear) if linear > 0.0 else SILENT_DB
AudioServer.set_bus_volume_db(idx, db)
func _load_sfx(key: StringName) -> AudioStream:
if _sfx_streams.has(key):
return _sfx_streams[key]
var path: String = String(SFX_FILES[key])
var stream: AudioStream = load(path)
if stream != null:
_sfx_streams[key] = stream
return stream
func _load_music(key: StringName) -> AudioStream:
if _music_streams.has(key):
return _music_streams[key]
var path: String = String(MUSIC_FILES[key])
var stream: AudioStream = load(path)
if stream != null:
# Loopable variants of the source tracks are pre-cut; tell the import
# layer to loop in case the .import wasn't flagged.
if stream is AudioStreamOggVorbis:
stream.loop = true
_music_streams[key] = stream
return stream
func _apply_phase_to_music(phase: StringName) -> void:
# Day / dawn / dusk all share the day loop; only night gets the night loop.
# Phase 20 may split dusk into a distinct track once we have one.
var key: StringName = &"night_loop" if phase == &"night" else &"day_loop"
play_music(key)
# ── signal handlers ─────────────────────────────────────────────────────────
func _on_pawn_took_damage(_pawn, _amount: float) -> void:
play_sfx(&"combat_hit")
func _on_storyteller_event_fired(_event) -> void:
play_sfx(&"ui_confirm")
func _on_alert_added(_severity: StringName, _text: String, _focus_tile: Vector2i) -> void:
play_sfx(&"ui_click")
func _on_clock_phase_changed(new_phase: StringName) -> void:
_apply_phase_to_music(new_phase)