diff --git a/HKMP/Game/Client/Entity/Component/MusicComponent.cs b/HKMP/Game/Client/Entity/Component/MusicComponent.cs
new file mode 100644
index 0000000..dc1cfbc
--- /dev/null
+++ b/HKMP/Game/Client/Entity/Component/MusicComponent.cs
@@ -0,0 +1,148 @@
+using System;
+using System.Collections.Generic;
+using Hkmp.Networking.Client;
+using Hkmp.Networking.Packet.Data;
+using Hkmp.Util;
+using HutongGames.PlayMaker.Actions;
+using Newtonsoft.Json;
+using Newtonsoft.Json.Converters;
+using UnityEngine;
+using UnityEngine.Audio;
+using Logger = Hkmp.Logging.Logger;
+
+namespace Hkmp.Game.Client.Entity.Component;
+
+// TODO: document all fields and methods
+///
+/// This component manages the music that plays for boss fights.
+internal class MusicComponent : EntityComponent {
+ ///
+ /// The file path of the embedded resource file for music data.
+ ///
+ private const string MusicDataFilePath = "Hkmp.Resource.music-data.json";
+
+ private static readonly List MusicCueDataList;
+ private static readonly List SnapshotDataList;
+
+ static MusicComponent() {
+ var dataPair = FileUtil.LoadObjectFromEmbeddedJson<
+ (List, List)
+ >(MusicDataFilePath);
+
+ MusicCueDataList = dataPair.Item1;
+ SnapshotDataList = dataPair.Item2;
+
+ On.PlayMakerFSM.OnEnable += OnFsmEnable;
+ }
+
+ public MusicComponent(
+ NetClient netClient,
+ ushort entityId,
+ HostClientPair gameObject
+ ) : base(netClient, entityId, gameObject) {
+ // TODO: register hooks for entering ApplyMusicCue and TransitionToAudioSnapshot actions
+ // TODO: these hooks should network changes in Music and Audio to the server if the player is scene host
+ }
+
+ ///
+ public override void InitializeHost() {
+ }
+
+ ///
+ public override void Update(EntityNetworkData data) {
+ // TODO: handle receiving Music and Audio updates from the server and applying them
+ }
+
+ ///
+ public override void Destroy() {
+ }
+
+ private static void OnFsmEnable(On.PlayMakerFSM.orig_OnEnable orig, PlayMakerFSM self) {
+ orig(self);
+
+ foreach (var state in self.FsmStates) {
+ foreach (var action in state.Actions) {
+ if (action is ApplyMusicCue applyMusicCue) {
+ var musicCue = applyMusicCue.musicCue.Value as MusicCue;
+ if (musicCue == null) {
+ continue;
+ }
+
+ Logger.Debug($"Found music cue '{musicCue.name}' in FSM '{self.Fsm.Name}', '{state.Name}'");
+
+ if (GetMusicCueData(
+ data => data.Name.Equals(musicCue.name),
+ out var musicCueData
+ )) {
+ Logger.Debug($" Adding to data with type: {musicCueData.Type}");
+ musicCueData.MusicCue = musicCue;
+ }
+ } else if (action is TransitionToAudioSnapshot snapshotAction) {
+ var snapshot = snapshotAction.snapshot.Value as AudioMixerSnapshot;
+ if (snapshot == null) {
+ continue;
+ }
+
+ Logger.Debug($"Found audio mixer snapshot '{snapshot.name}' in FSM '{self.Fsm.Name}', '{state.Name}'");
+ }
+ }
+ }
+ }
+
+ private static bool GetMusicCueData(Func predicate, out MusicCueData musicCueData) {
+ foreach (var data in MusicCueDataList) {
+ if (predicate.Invoke(data)) {
+ musicCueData = data;
+ return true;
+ }
+ }
+
+ musicCueData = null;
+ return false;
+ }
+
+ private class MusicCueData {
+ public MusicCueType Type { get; set; }
+ public string Name { get; set; }
+ public byte Index { get; set; }
+ [JsonIgnore]
+ public MusicCue MusicCue { get; set; }
+ }
+
+ private class AudioMixerSnapshotData {
+ public AudioMixerSnapshotType Type { get; set; }
+ public string Name { get; set; }
+ public byte Index { get; set; }
+ [JsonIgnore]
+ public AudioMixerSnapshot Snapshot { get; set; }
+ }
+
+ [JsonConverter(typeof(StringEnumConverter))]
+ private enum MusicCueType {
+ None,
+ FalseKnight,
+ Hornet,
+ GGHornet,
+ MantisLords,
+ SoulMaster,
+ SoulMaster2,
+ GGHeavy,
+ EnemyBattle,
+ DreamFight,
+ Hive,
+ HiveKnight,
+ DungDefender,
+ BrokenVessel,
+ Nosk,
+ TheHollowKnight,
+ Greenpath,
+ Waterways
+ }
+
+ [JsonConverter(typeof(StringEnumConverter))]
+ private enum AudioMixerSnapshotType {
+ Silent,
+ None,
+ Off,
+ }
+}
diff --git a/HKMP/HKMP.csproj b/HKMP/HKMP.csproj
index 76902f0..dc66cd4 100644
--- a/HKMP/HKMP.csproj
+++ b/HKMP/HKMP.csproj
@@ -23,6 +23,7 @@
+
diff --git a/HKMP/Resource/music-data.json b/HKMP/Resource/music-data.json
new file mode 100644
index 0000000..e9669f4
--- /dev/null
+++ b/HKMP/Resource/music-data.json
@@ -0,0 +1,104 @@
+{
+ "Item1": [
+ {
+ "Type": "None",
+ "Name": "None",
+ "Index": 0
+ },
+ {
+ "Type": "FalseKnight",
+ "Name": "Boss1",
+ "Index": 1
+ },
+ {
+ "Type": "Hornet",
+ "Name": "BossHornet",
+ "Index": 2
+ },
+ {
+ "Type": "GGHornet",
+ "Name": "GGHornet",
+ "Index": 3
+ },
+ {
+ "Type": "MantisLords",
+ "Name": "BossMantisLords",
+ "Index": 4
+ },
+ {
+ "Type": "SoulMaster",
+ "Name": "BossMageLord",
+ "Index": 5
+ },
+ {
+ "Type": "SoulMaster2",
+ "Name": "MageLord2",
+ "Index": 6
+ },
+ {
+ "Type": "GGHeavy",
+ "Name": "GG Heavy",
+ "Index": 7
+ },{
+ "Type": "DreamFight",
+ "Name": "DreamFight",
+ "Index": 8
+ },
+ {
+ "Type": "Hive",
+ "Name": "Hive",
+ "Index": 9
+ },
+ {
+ "Type": "HiveKnight",
+ "Name": "HiveKnight",
+ "Index": 10
+ },
+ {
+ "Type": "DungDefender",
+ "Name": "DungDefender",
+ "Index": 11
+ },
+ {
+ "Type": "BrokenVessel",
+ "Name": "BossIK",
+ "Index": 12
+ },
+ {
+ "Type": "Nosk",
+ "Name": "MimicSpider",
+ "Index": 13
+ },{
+ "Type": "TheHollowKnight",
+ "Name": "HKBattle",
+ "Index": 14
+ },
+ {
+ "Type": "Greenpath",
+ "Name": "Greenpath",
+ "Index": 15
+ },
+ {
+ "Type": "Waterways",
+ "Name": "Waterways",
+ "Index": 16
+ }
+ ],
+ "Item2": [
+ {
+ "Type": "Silent",
+ "Name": "Silent",
+ "Index": 0
+ },
+ {
+ "Type": "None",
+ "Name": "None",
+ "Index": 1
+ },
+ {
+ "Type": "Off",
+ "Name": "Off",
+ "Index": 2
+ }
+ ]
+}