diff --git a/_module/are/dbi_pc_rooms.are.json b/_module/are/dbi_pc_rooms.are.json new file mode 100644 index 0000000..b3b4c08 --- /dev/null +++ b/_module/are/dbi_pc_rooms.are.json @@ -0,0 +1,350 @@ +{ + "__data_type": "ARE ", + "ChanceLightning": { + "type": "int", + "value": 0 + }, + "ChanceRain": { + "type": "int", + "value": 0 + }, + "ChanceSnow": { + "type": "int", + "value": 0 + }, + "Comments": { + "type": "cexostring", + "value": "Dragonback Inn, Your Room" + }, + "Creator_ID": { + "type": "int", + "value": -1 + }, + "DayNightCycle": { + "type": "byte", + "value": 0 + }, + "Expansion_List": { + "type": "list", + "value": [] + }, + "Flags": { + "type": "dword", + "value": 1 + }, + "FogClipDist": { + "type": "float", + "value": 45.0 + }, + "Height": { + "type": "int", + "value": 2 + }, + "ID": { + "type": "int", + "value": -1 + }, + "IsNight": { + "type": "byte", + "value": 1 + }, + "LightingScheme": { + "type": "byte", + "value": 13 + }, + "LoadScreenID": { + "type": "word", + "value": 47 + }, + "ModListenCheck": { + "type": "int", + "value": 0 + }, + "ModSpotCheck": { + "type": "int", + "value": 0 + }, + "MoonAmbientColor": { + "type": "dword", + "value": 2960685 + }, + "MoonDiffuseColor": { + "type": "dword", + "value": 6457991 + }, + "MoonFogAmount": { + "type": "byte", + "value": 5 + }, + "MoonFogColor": { + "type": "dword", + "value": 0 + }, + "MoonShadows": { + "type": "byte", + "value": 0 + }, + "Name": { + "type": "cexolocstring", + "value": { + "0": "Dragonback Inn, Your Room" + } + }, + "NoRest": { + "type": "byte", + "value": 0 + }, + "OnEnter": { + "type": "resref", + "value": "" + }, + "OnExit": { + "type": "resref", + "value": "pc_room_onexit" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "PlayerVsPlayer": { + "type": "byte", + "value": 0 + }, + "ResRef": { + "type": "resref", + "value": "dbi_pc_rooms" + }, + "ShadowOpacity": { + "type": "byte", + "value": 60 + }, + "SkyBox": { + "type": "byte", + "value": 6 + }, + "SunAmbientColor": { + "type": "dword", + "value": 0 + }, + "SunDiffuseColor": { + "type": "dword", + "value": 0 + }, + "SunFogAmount": { + "type": "byte", + "value": 0 + }, + "SunFogColor": { + "type": "dword", + "value": 0 + }, + "SunShadows": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "DBI_PC_ROOMS" + }, + "Tile_List": { + "type": "list", + "value": [ + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 4 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 3 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 3 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 30 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 2 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 2 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 4 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 3 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 2 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 2 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 191 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 30 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 14 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 3 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 3 + } + } + ] + }, + "TileBrdrDisabled": { + "type": "byte", + "value": 0 + }, + "Tileset": { + "type": "resref", + "value": "tin01" + }, + "Version": { + "type": "dword", + "value": 18 + }, + "Width": { + "type": "int", + "value": 2 + }, + "WindPower": { + "type": "int", + "value": 0 + } +} diff --git a/_module/are/dragonbackinn.are.json b/_module/are/dragonbackinn.are.json index d7272d2..74435b9 100644 --- a/_module/are/dragonbackinn.are.json +++ b/_module/are/dragonbackinn.are.json @@ -14,7 +14,7 @@ }, "Comments": { "type": "cexostring", - "value": "" + "value": "Dragonback Inn" }, "Creator_ID": { "type": "int", @@ -50,7 +50,7 @@ }, "LightingScheme": { "type": "byte", - "value": 12 + "value": 13 }, "LoadScreenID": { "type": "word", @@ -66,11 +66,11 @@ }, "MoonAmbientColor": { "type": "dword", - "value": 3947580 + "value": 2960685 }, "MoonDiffuseColor": { "type": "dword", - "value": 11184810 + "value": 6457991 }, "MoonFogAmount": { "type": "byte", @@ -87,7 +87,7 @@ "Name": { "type": "cexolocstring", "value": { - "0": "DragonBack INN" + "0": "Dragonback Inn" } }, "NoRest": { @@ -175,6 +175,49 @@ "type": "int", "value": 58 }, + "Tile_MainLight1": { + "type": "byte", + "value": 30 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 0 + }, + "Tile_Orientation": { + "type": "int", + "value": 3 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 0 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 0 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 148 + }, "Tile_MainLight1": { "type": "byte", "value": 0 @@ -187,6 +230,92 @@ "type": "int", "value": 3 }, + "Tile_SrcLight1": { + "type": "byte", + "value": 0 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 0 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 147 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 0 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 0 + }, + "Tile_Orientation": { + "type": "int", + "value": 3 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 2 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 2 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 107 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 0 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 0 + }, + "Tile_Orientation": { + "type": "int", + "value": 1 + }, "Tile_SrcLight1": { "type": "byte", "value": 3 @@ -226,135 +355,6 @@ "type": "byte", "value": 14 }, - "Tile_Orientation": { - "type": "int", - "value": 3 - }, - "Tile_SrcLight1": { - "type": "byte", - "value": 3 - }, - "Tile_SrcLight2": { - "type": "byte", - "value": 3 - } - }, - { - "__struct_id": 1, - "Tile_AnimLoop1": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop2": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop3": { - "type": "byte", - "value": 1 - }, - "Tile_Height": { - "type": "int", - "value": 0 - }, - "Tile_ID": { - "type": "int", - "value": 147 - }, - "Tile_MainLight1": { - "type": "byte", - "value": 4 - }, - "Tile_MainLight2": { - "type": "byte", - "value": 0 - }, - "Tile_Orientation": { - "type": "int", - "value": 3 - }, - "Tile_SrcLight1": { - "type": "byte", - "value": 0 - }, - "Tile_SrcLight2": { - "type": "byte", - "value": 0 - } - }, - { - "__struct_id": 1, - "Tile_AnimLoop1": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop2": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop3": { - "type": "byte", - "value": 1 - }, - "Tile_Height": { - "type": "int", - "value": 0 - }, - "Tile_ID": { - "type": "int", - "value": 107 - }, - "Tile_MainLight1": { - "type": "byte", - "value": 4 - }, - "Tile_MainLight2": { - "type": "byte", - "value": 13 - }, - "Tile_Orientation": { - "type": "int", - "value": 1 - }, - "Tile_SrcLight1": { - "type": "byte", - "value": 3 - }, - "Tile_SrcLight2": { - "type": "byte", - "value": 3 - } - }, - { - "__struct_id": 1, - "Tile_AnimLoop1": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop2": { - "type": "byte", - "value": 1 - }, - "Tile_AnimLoop3": { - "type": "byte", - "value": 1 - }, - "Tile_Height": { - "type": "int", - "value": 0 - }, - "Tile_ID": { - "type": "int", - "value": 148 - }, - "Tile_MainLight1": { - "type": "byte", - "value": 4 - }, - "Tile_MainLight2": { - "type": "byte", - "value": 0 - }, "Tile_Orientation": { "type": "int", "value": 1 @@ -392,11 +392,11 @@ }, "Tile_MainLight1": { "type": "byte", - "value": 4 + "value": 30 }, "Tile_MainLight2": { "type": "byte", - "value": 0 + "value": 14 }, "Tile_Orientation": { "type": "int", @@ -439,7 +439,7 @@ }, "Tile_MainLight2": { "type": "byte", - "value": 0 + "value": 14 }, "Tile_Orientation": { "type": "int", @@ -447,11 +447,11 @@ }, "Tile_SrcLight1": { "type": "byte", - "value": 2 + "value": 0 }, "Tile_SrcLight2": { "type": "byte", - "value": 2 + "value": 0 } }, { @@ -521,11 +521,11 @@ }, "Tile_MainLight1": { "type": "byte", - "value": 0 + "value": 30 }, "Tile_MainLight2": { "type": "byte", - "value": 0 + "value": 13 }, "Tile_Orientation": { "type": "int", @@ -542,13 +542,17 @@ } ] }, + "TileBrdrDisabled": { + "type": "byte", + "value": 0 + }, "Tileset": { "type": "resref", "value": "tin01" }, "Version": { "type": "dword", - "value": 8 + "value": 12 }, "Width": { "type": "int", diff --git a/_module/are/hillsideinn.are.json b/_module/are/hillsideinn.are.json index 536fde2..8e93d39 100644 --- a/_module/are/hillsideinn.are.json +++ b/_module/are/hillsideinn.are.json @@ -327,13 +327,17 @@ } ] }, + "TileBrdrDisabled": { + "type": "byte", + "value": 0 + }, "Tileset": { "type": "resref", "value": "tin01" }, "Version": { "type": "dword", - "value": 44 + "value": 45 }, "Width": { "type": "int", diff --git a/_module/are/ifm_pc_rooms.are.json b/_module/are/ifm_pc_rooms.are.json new file mode 100644 index 0000000..42080d7 --- /dev/null +++ b/_module/are/ifm_pc_rooms.are.json @@ -0,0 +1,346 @@ +{ + "__data_type": "ARE ", + "ChanceLightning": { + "type": "int", + "value": 0 + }, + "ChanceRain": { + "type": "int", + "value": 0 + }, + "ChanceSnow": { + "type": "int", + "value": 0 + }, + "Comments": { + "type": "cexostring", + "value": "City of Baleas: Inn of the Flying Monkey - Your Room" + }, + "Creator_ID": { + "type": "int", + "value": -1 + }, + "DayNightCycle": { + "type": "byte", + "value": 0 + }, + "Expansion_List": { + "type": "list", + "value": [] + }, + "Flags": { + "type": "dword", + "value": 1 + }, + "FogClipDist": { + "type": "float", + "value": 45.0 + }, + "Height": { + "type": "int", + "value": 2 + }, + "ID": { + "type": "int", + "value": -1 + }, + "IsNight": { + "type": "byte", + "value": 1 + }, + "LightingScheme": { + "type": "byte", + "value": 13 + }, + "LoadScreenID": { + "type": "word", + "value": 47 + }, + "ModListenCheck": { + "type": "int", + "value": 0 + }, + "ModSpotCheck": { + "type": "int", + "value": 0 + }, + "MoonAmbientColor": { + "type": "dword", + "value": 2960685 + }, + "MoonDiffuseColor": { + "type": "dword", + "value": 6457991 + }, + "MoonFogAmount": { + "type": "byte", + "value": 5 + }, + "MoonFogColor": { + "type": "dword", + "value": 0 + }, + "MoonShadows": { + "type": "byte", + "value": 0 + }, + "Name": { + "type": "cexolocstring", + "value": { + "0": "City of Baleas: Inn of the Flying Monkey, Your Room" + } + }, + "NoRest": { + "type": "byte", + "value": 0 + }, + "OnEnter": { + "type": "resref", + "value": "" + }, + "OnExit": { + "type": "resref", + "value": "pc_room_onexit" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "PlayerVsPlayer": { + "type": "byte", + "value": 0 + }, + "ResRef": { + "type": "resref", + "value": "ifm_pc_rooms" + }, + "ShadowOpacity": { + "type": "byte", + "value": 60 + }, + "SkyBox": { + "type": "byte", + "value": 6 + }, + "SunAmbientColor": { + "type": "dword", + "value": 0 + }, + "SunDiffuseColor": { + "type": "dword", + "value": 0 + }, + "SunFogAmount": { + "type": "byte", + "value": 0 + }, + "SunFogColor": { + "type": "dword", + "value": 0 + }, + "SunShadows": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "IFM_PC_ROOMS" + }, + "Tile_List": { + "type": "list", + "value": [ + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 4 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 3 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 3 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 30 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 2 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 2 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 94 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 4 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 13 + }, + "Tile_Orientation": { + "type": "int", + "value": 3 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 2 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 2 + } + }, + { + "__struct_id": 1, + "Tile_AnimLoop1": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop2": { + "type": "byte", + "value": 1 + }, + "Tile_AnimLoop3": { + "type": "byte", + "value": 1 + }, + "Tile_Height": { + "type": "int", + "value": 0 + }, + "Tile_ID": { + "type": "int", + "value": 191 + }, + "Tile_MainLight1": { + "type": "byte", + "value": 30 + }, + "Tile_MainLight2": { + "type": "byte", + "value": 14 + }, + "Tile_Orientation": { + "type": "int", + "value": 0 + }, + "Tile_SrcLight1": { + "type": "byte", + "value": 3 + }, + "Tile_SrcLight2": { + "type": "byte", + "value": 3 + } + } + ] + }, + "Tileset": { + "type": "resref", + "value": "tin01" + }, + "Version": { + "type": "dword", + "value": 14 + }, + "Width": { + "type": "int", + "value": 2 + }, + "WindPower": { + "type": "int", + "value": 0 + } +} diff --git a/_module/are/trinitybank.are.json b/_module/are/trinitybank.are.json index 17c07e0..1aea635 100644 --- a/_module/are/trinitybank.are.json +++ b/_module/are/trinitybank.are.json @@ -843,13 +843,17 @@ } ] }, + "TileBrdrDisabled": { + "type": "byte", + "value": 0 + }, "Tileset": { "type": "resref", "value": "tic01" }, "Version": { "type": "dword", - "value": 16 + "value": 17 }, "Width": { "type": "int", diff --git a/_module/dlg/doorcv_dbi_room.dlg.json b/_module/dlg/doorcv_dbi_room.dlg.json new file mode 100644 index 0000000..024821e --- /dev/null +++ b/_module/dlg/doorcv_dbi_room.dlg.json @@ -0,0 +1,235 @@ +{ + "__data_type": "DLG ", + "DelayEntry": { + "type": "dword", + "value": 0 + }, + "DelayReply": { + "type": "dword", + "value": 0 + }, + "EndConverAbort": { + "type": "resref", + "value": "nw_walk_wp" + }, + "EndConversation": { + "type": "resref", + "value": "nw_walk_wp" + }, + "EntryList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "RepliesList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 1 + }, + "IsChild": { + "type": "byte", + "value": 0 + } + }, + { + "__struct_id": 1, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 0 + }, + "IsChild": { + "type": "byte", + "value": 0 + } + } + ] + }, + "Script": { + "type": "resref", + "value": "" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Speaker": { + "type": "cexostring", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "This door leads up to your room, would you like to go there now?" + } + } + } + ] + }, + "NumWords": { + "type": "dword", + "value": 26 + }, + "PreventZoomIn": { + "type": "byte", + "value": 0 + }, + "ReplyList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "EntriesList": { + "type": "list", + "value": [] + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "Script": { + "type": "resref", + "value": "" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "No, not just yet." + } + } + }, + { + "__struct_id": 1, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "EntriesList": { + "type": "list", + "value": [] + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "Script": { + "type": "resref", + "value": "at_ent_dbi_room" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "Yes, I think it's time for some rest." + } + } + } + ] + }, + "StartingList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 0 + } + } + ] + } +} diff --git a/_module/dlg/doorcv_inn_room.dlg.json b/_module/dlg/doorcv_inn_room.dlg.json new file mode 100644 index 0000000..06e76d2 --- /dev/null +++ b/_module/dlg/doorcv_inn_room.dlg.json @@ -0,0 +1,235 @@ +{ + "__data_type": "DLG ", + "DelayEntry": { + "type": "dword", + "value": 0 + }, + "DelayReply": { + "type": "dword", + "value": 0 + }, + "EndConverAbort": { + "type": "resref", + "value": "nw_walk_wp" + }, + "EndConversation": { + "type": "resref", + "value": "nw_walk_wp" + }, + "EntryList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "RepliesList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 1 + }, + "IsChild": { + "type": "byte", + "value": 0 + } + }, + { + "__struct_id": 1, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 0 + }, + "IsChild": { + "type": "byte", + "value": 0 + } + } + ] + }, + "Script": { + "type": "resref", + "value": "" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Speaker": { + "type": "cexostring", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "This door leads up to your room, would you like to go there now?" + } + } + } + ] + }, + "NumWords": { + "type": "dword", + "value": 26 + }, + "PreventZoomIn": { + "type": "byte", + "value": 0 + }, + "ReplyList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "EntriesList": { + "type": "list", + "value": [] + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "Script": { + "type": "resref", + "value": "" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "No, not just yet." + } + } + }, + { + "__struct_id": 1, + "ActionParams": { + "type": "list", + "value": [] + }, + "Animation": { + "type": "dword", + "value": 0 + }, + "AnimLoop": { + "type": "byte", + "value": 1 + }, + "Comment": { + "type": "cexostring", + "value": "" + }, + "Delay": { + "type": "dword", + "value": 4294967295 + }, + "EntriesList": { + "type": "list", + "value": [] + }, + "Quest": { + "type": "cexostring", + "value": "" + }, + "Script": { + "type": "resref", + "value": "at_enter_pc_room" + }, + "Sound": { + "type": "resref", + "value": "" + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "Yes, I think it's time for some rest." + } + } + } + ] + }, + "StartingList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "Active": { + "type": "resref", + "value": "" + }, + "ConditionParams": { + "type": "list", + "value": [] + }, + "Index": { + "type": "dword", + "value": 0 + } + } + ] + } +} diff --git a/_module/gic/dbi_pc_rooms.gic.json b/_module/gic/dbi_pc_rooms.gic.json new file mode 100644 index 0000000..2cc0f16 --- /dev/null +++ b/_module/gic/dbi_pc_rooms.gic.json @@ -0,0 +1,104 @@ +{ + "__data_type": "GIC ", + "Creature List": { + "type": "list", + "value": [] + }, + "Door List": { + "type": "list", + "value": [ + { + "__struct_id": 8, + "Comment": { + "type": "cexostring", + "value": "Strong Door" + } + } + ] + }, + "Encounter List": { + "type": "list", + "value": [] + }, + "List": { + "type": "list", + "value": [] + }, + "Placeable List": { + "type": "list", + "value": [ + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "volition's Bio Placeables by volition\r\n" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Low treasure script." + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Chair that one can sit on.\r\n\r\nCreated by Shir'le E. Illios\r\nshirle@drowwanderer.com\r\nhttp://chosen.drowwanderer.com\r\n" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "TCC - Interiors\r\nby The_NWNCCC" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + } + ] + }, + "SoundList": { + "type": "list", + "value": [] + }, + "StoreList": { + "type": "list", + "value": [] + }, + "TriggerList": { + "type": "list", + "value": [] + }, + "WaypointList": { + "type": "list", + "value": [] + } +} diff --git a/_module/gic/dragonbackinn.gic.json b/_module/gic/dragonbackinn.gic.json index 4bf2613..6264e95 100644 --- a/_module/gic/dragonbackinn.gic.json +++ b/_module/gic/dragonbackinn.gic.json @@ -61,7 +61,7 @@ "__struct_id": 8, "Comment": { "type": "cexostring", - "value": "" + "value": "\"Create Area\" Door" } } ] @@ -220,6 +220,13 @@ "type": "cexostring", "value": "" } + }, + { + "__struct_id": 5, + "Comment": { + "type": "cexostring", + "value": "On the Advanced tab, replace with whatever information you wish to appear on the Map of an area." + } } ] } diff --git a/_module/gic/ifm_pc_rooms.gic.json b/_module/gic/ifm_pc_rooms.gic.json new file mode 100644 index 0000000..99a163c --- /dev/null +++ b/_module/gic/ifm_pc_rooms.gic.json @@ -0,0 +1,118 @@ +{ + "__data_type": "GIC ", + "Creature List": { + "type": "list", + "value": [] + }, + "Door List": { + "type": "list", + "value": [ + { + "__struct_id": 8, + "Comment": { + "type": "cexostring", + "value": "Strong Door" + } + } + ] + }, + "Encounter List": { + "type": "list", + "value": [] + }, + "List": { + "type": "list", + "value": [] + }, + "Placeable List": { + "type": "list", + "value": [ + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "volition's Bio Placeables by volition\r\n" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Low treasure script." + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Chair that one can sit on.\r\n\r\nCreated by Shir'le E. Illios\r\nshirle@drowwanderer.com\r\nhttp://chosen.drowwanderer.com\r\n" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "TCC - Interiors\r\nby The_NWNCCC" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Source: Bathroom Fixtures by Lisa" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "Charlie's Oriental Placables v1.0 by Charlie\r\n" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } + } + ] + }, + "SoundList": { + "type": "list", + "value": [] + }, + "StoreList": { + "type": "list", + "value": [] + }, + "TriggerList": { + "type": "list", + "value": [] + }, + "WaypointList": { + "type": "list", + "value": [] + } +} diff --git a/_module/gic/trinitybank.gic.json b/_module/gic/trinitybank.gic.json index c07ccde..d864d91 100644 --- a/_module/gic/trinitybank.gic.json +++ b/_module/gic/trinitybank.gic.json @@ -216,6 +216,13 @@ "type": "cexostring", "value": "Urn" } + }, + { + "__struct_id": 9, + "Comment": { + "type": "cexostring", + "value": "" + } } ] }, diff --git a/_module/git/cheatcentral.git.json b/_module/git/cheatcentral.git.json index e5a37a7..944f862 100644 --- a/_module/git/cheatcentral.git.json +++ b/_module/git/cheatcentral.git.json @@ -375,6 +375,82 @@ "type": "resref", "value": "dmrobes" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 24 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -697,6 +773,82 @@ "type": "resref", "value": "aarcl015" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 6 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1152,6 +1304,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.7572088241577148 @@ -1502,6 +1730,82 @@ "type": "resref", "value": "dmrobes" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 24 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1824,6 +2128,82 @@ "type": "resref", "value": "nw_aarcl006" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 6 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -2279,6 +2659,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": -0.4929004311561585 @@ -2629,6 +3085,82 @@ "type": "resref", "value": "dmrobes" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 24 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -2951,6 +3483,82 @@ "type": "resref", "value": "nw_aarcl006" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 6 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -3406,6 +4014,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.3136817514896393 @@ -3756,6 +4440,82 @@ "type": "resref", "value": "dmrobes" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 24 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 25 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4078,6 +4838,82 @@ "type": "resref", "value": "nw_aarcl006" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 6 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 17 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 9 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4533,6 +5369,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -5547,6 +6459,10 @@ "type": "resref", "value": "beltofjagang" }, + "xModelPart1": { + "type": "word", + "value": 7 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -5901,6 +6817,10 @@ "type": "resref", "value": "grendelbelt" }, + "xModelPart1": { + "type": "word", + "value": 5 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -6317,6 +7237,10 @@ "type": "resref", "value": "beltofthecenobit" }, + "xModelPart1": { + "type": "word", + "value": 8 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7693,6 +8617,82 @@ "type": "resref", "value": "celolasrobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 16 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 22 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 5 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 9 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -8852,6 +9852,82 @@ "type": "resref", "value": "robeoflegends" }, + "xArmorPart_Belt": { + "type": "word", + "value": 16 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 22 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 5 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 9 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -9703,6 +10779,82 @@ "type": "resref", "value": "ancientshogunarm" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 3 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 16 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 26 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 3 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 3 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 8 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 16 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 6 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -10521,6 +11673,82 @@ "type": "resref", "value": "cainarmor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 18 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 11 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 16 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 10 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 6 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 18 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 11 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 16 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 10 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 37 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -10897,6 +12125,10 @@ "type": "resref", "value": "ancientshogunhel" }, + "xModelPart1": { + "type": "word", + "value": 28 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -11457,6 +12689,10 @@ "type": "resref", "value": "ancientroguehelm" }, + "xModelPart1": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -12110,6 +13346,10 @@ "type": "resref", "value": "cainhelm" }, + "xModelPart1": { + "type": "word", + "value": 23 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -12835,6 +14075,82 @@ "type": "resref", "value": "lithiumlight" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 3 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 22 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 3 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 5 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 50 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -13839,6 +15155,82 @@ "type": "resref", "value": "leathersoftheund" }, + "xArmorPart_Belt": { + "type": "word", + "value": 11 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 7 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -14440,6 +15832,82 @@ "type": "resref", "value": "bladeslingerarmo" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 13 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 18 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 4 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 13 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 13 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -14883,6 +16351,10 @@ "type": "resref", "value": "shieldofthewarri" }, + "xModelPart1": { + "type": "word", + "value": 41 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -15299,6 +16771,10 @@ "type": "resref", "value": "beltofthecenobit" }, + "xModelPart1": { + "type": "word", + "value": 8 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -15556,6 +17032,10 @@ "type": "resref", "value": "beltofspellcraft" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16030,6 +17510,10 @@ "type": "resref", "value": "beltofjagang" }, + "xModelPart1": { + "type": "word", + "value": 7 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16384,6 +17868,10 @@ "type": "resref", "value": "grendelbelt" }, + "xModelPart1": { + "type": "word", + "value": 5 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16835,6 +18323,18 @@ "type": "resref", "value": "trinitychampi002" }, + "xModelPart1": { + "type": "word", + "value": 41 + }, + "xModelPart2": { + "type": "word", + "value": 23 + }, + "xModelPart3": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17290,6 +18790,18 @@ "type": "resref", "value": "elevenmageboots" }, + "xModelPart1": { + "type": "word", + "value": 31 + }, + "xModelPart2": { + "type": "word", + "value": 32 + }, + "xModelPart3": { + "type": "word", + "value": 32 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17590,6 +19102,18 @@ "type": "resref", "value": "hardyboots" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 12 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18243,6 +19767,10 @@ "type": "resref", "value": "awdrowcloak" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18555,6 +20083,10 @@ "type": "resref", "value": "shoguncloak" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18905,6 +20437,10 @@ "type": "resref", "value": "awapprenticebrac" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19441,6 +20977,10 @@ "type": "resref", "value": "mysticbracer" }, + "xModelPart1": { + "type": "word", + "value": 4 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19698,6 +21238,10 @@ "type": "resref", "value": "dwarvenbracers" }, + "xModelPart1": { + "type": "word", + "value": 5 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20110,6 +21654,10 @@ "type": "resref", "value": "boradhammerbrace" }, + "xModelPart1": { + "type": "word", + "value": 6 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20866,6 +22414,82 @@ "type": "resref", "value": "awdrowassassinar" }, + "xArmorPart_Belt": { + "type": "word", + "value": 12 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 13 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 6 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 21 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 12 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 5 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 13 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 6 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 21 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 12 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21364,6 +22988,10 @@ "type": "resref", "value": "legionroguehelm" }, + "xModelPart1": { + "type": "word", + "value": 4 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21900,6 +23528,10 @@ "type": "resref", "value": "ancientshogungau" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22660,6 +24292,82 @@ "type": "resref", "value": "masterassassintu" }, + "xArmorPart_Belt": { + "type": "word", + "value": 3 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 3 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 14 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 22 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 3 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 14 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 6 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23183,6 +24891,18 @@ "type": "resref", "value": "lithiumbastard" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 44 + }, + "xModelPart3": { + "type": "word", + "value": 14 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23543,6 +25263,18 @@ "type": "resref", "value": "ironclawbastard" }, + "xModelPart1": { + "type": "word", + "value": 33 + }, + "xModelPart2": { + "type": "word", + "value": 32 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23996,6 +25728,18 @@ "type": "resref", "value": "trinitychampswor" }, + "xModelPart1": { + "type": "word", + "value": 23 + }, + "xModelPart2": { + "type": "word", + "value": 42 + }, + "xModelPart3": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24356,6 +26100,18 @@ "type": "resref", "value": "irongoblindagger" }, + "xModelPart1": { + "type": "word", + "value": 62 + }, + "xModelPart2": { + "type": "word", + "value": 13 + }, + "xModelPart3": { + "type": "word", + "value": 61 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24523,6 +26279,10 @@ "type": "resref", "value": "firedart" }, + "xModelPart1": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24842,6 +26602,10 @@ "type": "resref", "value": "starsofholyness" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25262,6 +27026,18 @@ "type": "resref", "value": "awdrowbow" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, + "xModelPart2": { + "type": "word", + "value": 54 + }, + "xModelPart3": { + "type": "word", + "value": 74 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25713,6 +27489,18 @@ "type": "resref", "value": "awdrowhalbred" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26040,6 +27828,18 @@ "type": "resref", "value": "desertkukri" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 14 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26398,6 +28198,18 @@ "type": "resref", "value": "awcelolaswhip" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, + "xModelPart2": { + "type": "word", + "value": 11 + }, + "xModelPart3": { + "type": "word", + "value": 11 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26975,6 +28787,18 @@ "type": "resref", "value": "item006" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 11 + }, + "xModelPart3": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27457,6 +29281,18 @@ "type": "resref", "value": "awdriderscythe" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 24 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27850,6 +29686,18 @@ "type": "resref", "value": "dragondbleaxe" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 12 + }, + "xModelPart3": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28210,6 +30058,18 @@ "type": "resref", "value": "atlmace" }, + "xModelPart1": { + "type": "word", + "value": 32 + }, + "xModelPart2": { + "type": "word", + "value": 22 + }, + "xModelPart3": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28789,6 +30649,18 @@ "type": "resref", "value": "cainhammer" }, + "xModelPart1": { + "type": "word", + "value": 64 + }, + "xModelPart2": { + "type": "word", + "value": 64 + }, + "xModelPart3": { + "type": "word", + "value": 54 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29304,6 +31176,18 @@ "type": "resref", "value": "lithiumbastard" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 44 + }, + "xModelPart3": { + "type": "word", + "value": 14 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29757,6 +31641,18 @@ "type": "resref", "value": "trinitychampswor" }, + "xModelPart1": { + "type": "word", + "value": 23 + }, + "xModelPart2": { + "type": "word", + "value": 42 + }, + "xModelPart3": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30332,6 +32228,18 @@ "type": "resref", "value": "awdrowdagger" }, + "xModelPart1": { + "type": "word", + "value": 43 + }, + "xModelPart2": { + "type": "word", + "value": 33 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30690,6 +32598,18 @@ "type": "resref", "value": "jimmyssword" }, + "xModelPart1": { + "type": "word", + "value": 72 + }, + "xModelPart2": { + "type": "word", + "value": 72 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31172,6 +33092,18 @@ "type": "resref", "value": "cainlongsword" }, + "xModelPart1": { + "type": "word", + "value": 84 + }, + "xModelPart2": { + "type": "word", + "value": 84 + }, + "xModelPart3": { + "type": "word", + "value": 74 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31592,6 +33524,18 @@ "type": "resref", "value": "awdrowrapier" }, + "xModelPart1": { + "type": "word", + "value": 41 + }, + "xModelPart2": { + "type": "word", + "value": 44 + }, + "xModelPart3": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31987,6 +33931,18 @@ "type": "resref", "value": "dirthrrapier" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32413,6 +34369,18 @@ "type": "resref", "value": "wdarkqueenrapier" }, + "xModelPart1": { + "type": "word", + "value": 44 + }, + "xModelPart2": { + "type": "word", + "value": 24 + }, + "xModelPart3": { + "type": "word", + "value": 24 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32957,6 +34925,18 @@ "type": "resref", "value": "dragonmonkstaff" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 14 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33473,6 +35453,18 @@ "type": "resref", "value": "cainwaraxe" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 84 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34048,6 +36040,18 @@ "type": "resref", "value": "ironclawgreat001" }, + "xModelPart1": { + "type": "word", + "value": 44 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34851,6 +36855,18 @@ "type": "resref", "value": "awdrowstaff" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, + "xModelPart2": { + "type": "word", + "value": 24 + }, + "xModelPart3": { + "type": "word", + "value": 64 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34994,6 +37010,18 @@ "type": "resref", "value": "resrod" }, + "xModelPart1": { + "type": "word", + "value": 22 + }, + "xModelPart2": { + "type": "word", + "value": 32 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35344,6 +37372,10 @@ "type": "resref", "value": "ancientshogunamm" }, + "xModelPart1": { + "type": "word", + "value": 4 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35756,6 +37788,10 @@ "type": "resref", "value": "jerkinsamulet" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36168,6 +38204,10 @@ "type": "resref", "value": "awamuletofthemin" }, + "xModelPart1": { + "type": "word", + "value": 28 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36673,6 +38713,10 @@ "type": "resref", "value": "mindflayerammy" }, + "xModelPart1": { + "type": "word", + "value": 28 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37273,6 +39317,10 @@ "type": "resref", "value": "awringofveraplay" }, + "xModelPart1": { + "type": "word", + "value": 17 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37623,6 +39671,10 @@ "type": "resref", "value": "banditring" }, + "xModelPart1": { + "type": "word", + "value": 16 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37756,6 +39808,10 @@ "type": "resref", "value": "angelring" }, + "xModelPart1": { + "type": "word", + "value": 27 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37984,6 +40040,10 @@ "type": "resref", "value": "resistring" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38396,6 +40456,10 @@ "type": "resref", "value": "jerkinsamulet" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38808,6 +40872,10 @@ "type": "resref", "value": "ringofthehealer" }, + "xModelPart1": { + "type": "word", + "value": 3 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39654,6 +41722,10 @@ "type": "resref", "value": "ringofthemystic" }, + "xModelPart1": { + "type": "word", + "value": 24 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39932,7 +42004,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -39942,6 +42014,10 @@ "type": "resref", "value": "ringoflight" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40145,6 +42221,18 @@ "type": "resref", "value": "dmshelper" }, + "xModelPart1": { + "type": "word", + "value": 34 + }, + "xModelPart2": { + "type": "word", + "value": 34 + }, + "xModelPart3": { + "type": "word", + "value": 62 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40286,6 +42374,18 @@ "type": "resref", "value": "emotewand" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 22 + }, + "xModelPart3": { + "type": "word", + "value": 42 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40431,6 +42531,18 @@ "type": "resref", "value": "kotshelper001" }, + "xModelPart1": { + "type": "word", + "value": 33 + }, + "xModelPart2": { + "type": "word", + "value": 42 + }, + "xModelPart3": { + "type": "word", + "value": 42 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40576,6 +42688,18 @@ "type": "resref", "value": "autofollow" }, + "xModelPart1": { + "type": "word", + "value": 14 + }, + "xModelPart2": { + "type": "word", + "value": 14 + }, + "xModelPart3": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40720,6 +42844,10 @@ "type": "resref", "value": "firebombnpc" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40856,6 +42984,10 @@ "type": "resref", "value": "firebombnpc001" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40989,6 +43121,10 @@ "type": "resref", "value": "xx_dragonport" }, + "xModelPart1": { + "type": "word", + "value": 48 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41122,6 +43258,10 @@ "type": "resref", "value": "xx_dwarfport" }, + "xModelPart1": { + "type": "word", + "value": 48 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41255,6 +43395,10 @@ "type": "resref", "value": "xx_nordicport" }, + "xModelPart1": { + "type": "word", + "value": 48 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41388,6 +43532,10 @@ "type": "resref", "value": "xx_manyport" }, + "xModelPart1": { + "type": "word", + "value": 48 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41525,6 +43673,10 @@ "type": "resref", "value": "xx_testreward" }, + "xModelPart1": { + "type": "word", + "value": 85 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41658,6 +43810,10 @@ "type": "resref", "value": "xx_trinport" }, + "xModelPart1": { + "type": "word", + "value": 44 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41759,6 +43915,10 @@ "type": "resref", "value": "handsoftime002" }, + "xModelPart1": { + "type": "word", + "value": 124 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41892,6 +44052,10 @@ "type": "resref", "value": "xx_tempcain" }, + "xModelPart1": { + "type": "word", + "value": 81 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42025,6 +44189,10 @@ "type": "resref", "value": "banditstools" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42170,6 +44338,18 @@ "type": "resref", "value": "keyofcheating" }, + "xModelPart1": { + "type": "word", + "value": 41 + }, + "xModelPart2": { + "type": "word", + "value": 11 + }, + "xModelPart3": { + "type": "word", + "value": 61 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42303,6 +44483,10 @@ "type": "resref", "value": "xx_tempcain" }, + "xModelPart1": { + "type": "word", + "value": 81 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42440,6 +44624,10 @@ "type": "resref", "value": "cs_takegold" }, + "xModelPart1": { + "type": "word", + "value": 62 + }, "XOrientation": { "type": "float", "value": 0.0 diff --git a/_module/git/dbi_pc_rooms.git.json b/_module/git/dbi_pc_rooms.git.json new file mode 100644 index 0000000..b3afd24 --- /dev/null +++ b/_module/git/dbi_pc_rooms.git.json @@ -0,0 +1,2131 @@ +{ + "__data_type": "GIT ", + "AreaProperties": { + "__struct_id": 100, + "type": "struct", + "value": { + "__struct_id": 100, + "AmbientSndDay": { + "type": "int", + "value": 72 + }, + "AmbientSndDayVol": { + "type": "int", + "value": 32 + }, + "AmbientSndNight": { + "type": "int", + "value": 107 + }, + "AmbientSndNitVol": { + "type": "int", + "value": 12 + }, + "EnvAudio": { + "type": "int", + "value": 2 + }, + "MusicBattle": { + "type": "int", + "value": 40 + }, + "MusicDay": { + "type": "int", + "value": 25 + }, + "MusicDelay": { + "type": "int", + "value": 45000 + }, + "MusicNight": { + "type": "int", + "value": 25 + } + } + }, + "Creature List": { + "type": "list", + "value": [] + }, + "Door List": { + "type": "list", + "value": [ + { + "__struct_id": 8, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 0 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "The only exit from your room at the Dragonback Inn.", + "id": 9078 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "GenericType_New": { + "type": "dword", + "value": 12 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "LinkedTo": { + "type": "cexostring", + "value": "DBI_ROOMS2INN" + }, + "LinkedToFlags": { + "type": "byte", + "value": 2 + }, + "LoadScreenID": { + "type": "word", + "value": 42 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Door", + "id": 5349 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "se_door_death" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnFailToOpen": { + "type": "resref", + "value": "door_fail2open" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "ra_close_door" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 0 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "dbi_rooms2inn" + }, + "TemplateResRef": { + "type": "resref", + "value": "x3_door_wood002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 15.0 + }, + "Y": { + "type": "float", + "value": 20.0 + }, + "Z": { + "type": "float", + "value": 0.0 + } + } + ] + }, + "Encounter List": { + "type": "list", + "value": [] + }, + "List": { + "type": "list", + "value": [] + }, + "Placeable List": { + "type": "list", + "value": [ + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 286 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.7853981852531433 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16813643 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Fluffy Bed", + "id": 16811120 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 0 + }, + "PortraitId": { + "type": "word", + "value": 773 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_BED002" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_bed002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 16.78469085693359 + }, + "Y": { + "type": "float", + "value": 13.20924949645996 + }, + "Z": { + "type": "float", + "value": 0.009990006685256958 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 0 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.570796251296997 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "This looks like a safe place to store your gear.", + "id": 14548 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Armoire", + "id": 14549 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "nui_f_storage" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 358 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "Armoire" + }, + "TemplateResRef": { + "type": "resref", + "value": "plc_armoire" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.55000019073486 + }, + "Y": { + "type": "float", + "value": 16.82999992370606 + }, + "Z": { + "type": "float", + "value": 0.0 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 447 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.8590291738510132 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "You see yourself in the mirror.", + "id": 16810932 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Mirror", + "id": 16812334 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "sob_examine002" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 930 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_MIRROR005" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_mirror005" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.54697418212891 + }, + "Y": { + "type": "float", + "value": 18.48270606994629 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 12 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": -1.570796251296997 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 14607 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 0 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 5655 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "nw_o2_generallow" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "nw_o2_generallow" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 370 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "Desk" + }, + "TemplateResRef": { + "type": "resref", + "value": "plc_desk" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 18.48391151428223 + }, + "Y": { + "type": "float", + "value": 17.2309455871582 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 179 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.006291270256043 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "It is a simple chair but the grace of its lines speaks to the quality of its craftmanship.", + "id": 14593 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Chair", + "id": 5822 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "sei_sit" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 537 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "Chair" + }, + "TemplateResRef": { + "type": "resref", + "value": "chair001" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 17.41463851928711 + }, + "Y": { + "type": "float", + "value": 17.40582275390625 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 104 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 2.012582778930664 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16813707 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Tub", + "id": 16813470 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 462 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_TUB002" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_tub002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.66264915466309 + }, + "Y": { + "type": "float", + "value": 11.96749687194824 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 157 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": -3.141592264175415 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "A simple throw rug adorns the floor, faded and worn with use.", + "id": 16813674 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Threshold", + "id": 16813108 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 515 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "DBI_ROOM_MAT" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_throwrug058" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 15.05000019073486 + }, + "Y": { + "type": "float", + "value": 19.02000045776367 + }, + "Z": { + "type": "float", + "value": 0.0 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 15177 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.993602986962655e-019 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "This rug is beautifully woven, with intricate patterns that draw you in more the closer you look." + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Rug, Oriental - Green 1" + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 0 + }, + "PortraitId": { + "type": "word", + "value": 515 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "tm_pl_rugorigrn1" + }, + "TemplateResRef": { + "type": "resref", + "value": "tm_pl_rugorigrn1" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 14.95936584472656 + }, + "Y": { + "type": "float", + "value": 15.52633380889893 + }, + "Z": { + "type": "float", + "value": 0.009990006685256958 + } + } + ] + }, + "SoundList": { + "type": "list", + "value": [] + }, + "StoreList": { + "type": "list", + "value": [] + }, + "TriggerList": { + "type": "list", + "value": [] + }, + "WaypointList": { + "type": "list", + "value": [] + } +} diff --git a/_module/git/dragonbackinn.git.json b/_module/git/dragonbackinn.git.json index 56a68ee..632c900 100644 --- a/_module/git/dragonbackinn.git.json +++ b/_module/git/dragonbackinn.git.json @@ -35,7 +35,7 @@ }, "MusicDelay": { "type": "int", - "value": 90000 + "value": 45000 }, "MusicNight": { "type": "int", @@ -375,6 +375,82 @@ "type": "resref", "value": "barmaid" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 9 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 26 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 9 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -898,6 +974,82 @@ "type": "byte", "value": 10 }, + "xAppearance_Head": { + "type": "word", + "value": 2 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.3598990440368652 @@ -1248,6 +1400,82 @@ "type": "resref", "value": "nw_cloth027" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 6 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 37 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 6 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 39 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1813,6 +2041,82 @@ "type": "byte", "value": 10 }, + "xAppearance_Head": { + "type": "word", + "value": 2 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": -0.5758079290390015 @@ -4057,7 +4361,7 @@ }, "Bearing": { "type": "float", - "value": 1.570794224739075 + "value": 1.570796251296997 }, "CloseLockDC": { "type": "byte", @@ -4065,19 +4369,22 @@ }, "Conversation": { "type": "resref", - "value": "" + "value": "doorcv_dbi_room" }, "CurrentHP": { "type": "short", - "value": 10 + "value": 15 }, "Description": { "type": "cexolocstring", - "value": {} + "value": { + "0": "This door leads to the inn rooms above.", + "id": 9078 + } }, "DisarmDC": { "type": "byte", - "value": 0 + "value": 15 }, "Faction": { "type": "dword", @@ -4085,11 +4392,11 @@ }, "Fort": { "type": "byte", - "value": 5 + "value": 16 }, "GenericType_New": { "type": "dword", - "value": 0 + "value": 12 }, "Hardness": { "type": "byte", @@ -4097,7 +4404,7 @@ }, "HP": { "type": "short", - "value": 10 + "value": 15 }, "Interruptable": { "type": "byte", @@ -4109,7 +4416,7 @@ }, "KeyRequired": { "type": "byte", - "value": 0 + "value": 1 }, "LinkedTo": { "type": "cexostring", @@ -4134,7 +4441,8 @@ "LocName": { "type": "cexolocstring", "value": { - "0": "BasicWoodDoorLocked" + "0": "Inn Rooms", + "id": 5349 } }, "OnClick": { @@ -4151,7 +4459,7 @@ }, "OnDeath": { "type": "resref", - "value": "" + "value": "x2_door_death" }, "OnDisarm": { "type": "resref", @@ -4159,7 +4467,7 @@ }, "OnFailToOpen": { "type": "resref", - "value": "" + "value": "cv_inn_door" }, "OnHeartbeat": { "type": "resref", @@ -4175,7 +4483,7 @@ }, "OnOpen": { "type": "resref", - "value": "2_door_close_10" + "value": "" }, "OnSpellCastAt": { "type": "resref", @@ -4195,7 +4503,7 @@ }, "OpenLockDC": { "type": "byte", - "value": 200 + "value": 18 }, "Plot": { "type": "byte", @@ -4203,7 +4511,7 @@ }, "PortraitId": { "type": "word", - "value": 558 + "value": 2624 }, "Ref": { "type": "byte", @@ -4211,11 +4519,11 @@ }, "Tag": { "type": "cexostring", - "value": "BasicWoodDoorLocked" + "value": "dbi_inn2pcrooms" }, "TemplateResRef": { "type": "resref", - "value": "basicwooddoorloc" + "value": "inn2pcrooms" }, "TrapDetectable": { "type": "byte", @@ -4375,6 +4683,18 @@ "type": "resref", "value": "nw_it_mpotion021" }, + "xModelPart1": { + "type": "word", + "value": 56 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7444,6 +7764,71 @@ "type": "float", "value": 0.004569292068481445 } + }, + { + "__struct_id": 5, + "Appearance": { + "type": "byte", + "value": 1 + }, + "Description": { + "type": "cexolocstring", + "value": {} + }, + "HasMapNote": { + "type": "byte", + "value": 1 + }, + "LinkedTo": { + "type": "cexostring", + "value": "" + }, + "LocalizedName": { + "type": "cexolocstring", + "value": { + "0": "DBI_ROOMS2INN", + "id": 14814 + } + }, + "MapNote": { + "type": "cexolocstring", + "value": { + "0": "Rooms", + "id": 14815 + } + }, + "MapNoteEnabled": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "DBI_ROOMS2INN" + }, + "TemplateResRef": { + "type": "resref", + "value": "nw_mapnote001" + }, + "XOrientation": { + "type": "float", + "value": 1.0 + }, + "XPosition": { + "type": "float", + "value": 8.031503677368164 + }, + "YOrientation": { + "type": "float", + "value": 7.549790126404332e-008 + }, + "YPosition": { + "type": "float", + "value": 25.09390449523926 + }, + "ZPosition": { + "type": "float", + "value": 0.002738118171691895 + } } ] } diff --git a/_module/git/hillsideinn.git.json b/_module/git/hillsideinn.git.json index b995f26..2de7c02 100644 --- a/_module/git/hillsideinn.git.json +++ b/_module/git/hillsideinn.git.json @@ -311,7 +311,7 @@ }, "Cost": { "type": "dword", - "value": 2791716 + "value": 1388876 }, "Cursed": { "type": "byte", @@ -556,7 +556,7 @@ }, "CostValue": { "type": "word", - "value": 4 + "value": 3 }, "Param1": { "type": "byte", @@ -587,7 +587,7 @@ }, "CostValue": { "type": "word", - "value": 4 + "value": 3 }, "Param1": { "type": "byte", @@ -637,99 +637,6 @@ "value": 1 } }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 37 - }, - "Subtype": { - "type": "word", - "value": 6 - } - }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 37 - }, - "Subtype": { - "type": "word", - "value": 1 - } - }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 37 - }, - "Subtype": { - "type": "word", - "value": 7 - } - }, { "__struct_id": 0, "ChanceAppear": { @@ -893,16 +800,92 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", - "value": "ReinforcedHeavyArmor" + "value": "REINFORCEDHEAVYARMOR" }, "TemplateResRef": { "type": "resref", "value": "reinforcedarmor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 8 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 9 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1321,6 +1304,82 @@ "type": "resref", "value": "nw_mcloth011" }, + "xArmorPart_Belt": { + "type": "word", + "value": 9 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 18 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 3 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 9 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 18 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 50 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1776,6 +1835,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 21 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": -0.995184600353241 @@ -3878,6 +4013,82 @@ "type": "resref", "value": "cloakoftheshadow" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 26 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4512,6 +4723,82 @@ "type": "resref", "value": "femaledruidwarpi" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 1 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 1 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 28 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 1 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 1 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 5 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4633,7 +4920,7 @@ }, "Cost": { "type": "dword", - "value": 1934461 + "value": 1473217 }, "Cursed": { "type": "byte", @@ -5351,7 +5638,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -5361,6 +5648,82 @@ "type": "resref", "value": "mysticrobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 13 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 3 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -5482,7 +5845,7 @@ }, "Cost": { "type": "dword", - "value": 1975581 + "value": 1387377 }, "Cursed": { "type": "byte", @@ -5952,7 +6315,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -5962,6 +6325,82 @@ "type": "resref", "value": "reinforcedrobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 13 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 14 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 31 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 14 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -6083,7 +6522,7 @@ }, "Cost": { "type": "dword", - "value": 3277276 + "value": 2792859 }, "Cursed": { "type": "byte", @@ -6708,7 +7147,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -6718,6 +7157,82 @@ "type": "resref", "value": "robesofthehighwi" }, + "xArmorPart_Belt": { + "type": "word", + "value": 14 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 16 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 18 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 10 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 16 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 50 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7350,6 +7865,82 @@ "type": "resref", "value": "robesoftheperfor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 8 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 32 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 8 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 7 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7951,6 +8542,82 @@ "type": "resref", "value": "masterworkarmor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 8 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 8 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 9 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 8 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 23 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -8072,7 +8739,7 @@ }, "Cost": { "type": "dword", - "value": 1977080 + "value": 1388876 }, "Cursed": { "type": "byte", @@ -8573,7 +9240,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -8583,6 +9250,82 @@ "type": "resref", "value": "reinforcedarmor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 8 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 9 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 11 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 13 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -9019,6 +9762,10 @@ "type": "resref", "value": "masterworkhelm" }, + "xModelPart1": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -9610,6 +10357,10 @@ "type": "resref", "value": "mystichelm" }, + "xModelPart1": { + "type": "word", + "value": 28 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -10046,6 +10797,10 @@ "type": "resref", "value": "reinforcedhelm" }, + "xModelPart1": { + "type": "word", + "value": 17 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -10523,6 +11278,82 @@ "type": "resref", "value": "armorofseduction" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 9 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 1 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 28 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 9 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 1 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 10 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -11095,6 +11926,82 @@ "type": "resref", "value": "armorofthorns" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 13 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 30 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 13 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 28 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -11696,6 +12603,82 @@ "type": "resref", "value": "naturescloak" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 15 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 26 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 16 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 15 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -11817,7 +12800,7 @@ }, "Cost": { "type": "dword", - "value": 1975595 + "value": 1387391 }, "Cursed": { "type": "byte", @@ -12318,7 +13301,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -12328,6 +13311,82 @@ "type": "resref", "value": "reinforcedarmorl" }, + "xArmorPart_Belt": { + "type": "word", + "value": 14 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 6 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 5 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 6 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -12836,6 +13895,82 @@ "type": "resref", "value": "roguesarmor" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 14 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 3 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 26 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 14 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 6 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 6 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 15 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 20 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -13561,6 +14696,82 @@ "type": "resref", "value": "suitofthearchane" }, + "xArmorPart_Belt": { + "type": "word", + "value": 15 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 6 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 3 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 8 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 15 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 3 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -13682,7 +14893,7 @@ }, "Cost": { "type": "dword", - "value": 1975680 + "value": 1387476 }, "Cursed": { "type": "byte", @@ -14183,7 +15394,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -14193,6 +15404,82 @@ "type": "resref", "value": "reinforcedarmorm" }, + "xArmorPart_Belt": { + "type": "word", + "value": 10 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 23 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 6 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 9 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 5 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 15 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 5 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 12 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 23 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 6 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 4 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -14574,6 +15861,10 @@ "type": "resref", "value": "reinforcedtowers" }, + "xModelPart1": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -14920,6 +16211,10 @@ "type": "resref", "value": "reinforcedlarges" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -15177,6 +16472,10 @@ "type": "resref", "value": "beltofspellcraft" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -15544,7 +16843,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -15554,6 +16853,10 @@ "type": "resref", "value": "reinforcedbelt" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -15881,6 +17184,18 @@ "type": "resref", "value": "naturesfootsteps" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 41 + }, + "xModelPart3": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16208,6 +17523,18 @@ "type": "resref", "value": "reinforcedboots" }, + "xModelPart1": { + "type": "word", + "value": 21 + }, + "xModelPart2": { + "type": "word", + "value": 13 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16620,6 +17947,10 @@ "type": "resref", "value": "reinforcedbracer" }, + "xModelPart1": { + "type": "word", + "value": 3 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16994,6 +18325,10 @@ "type": "resref", "value": "cloakofthehills" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17368,6 +18703,10 @@ "type": "resref", "value": "reinforcedcloak" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17594,6 +18933,10 @@ "type": "resref", "value": "reinforcedgloves" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17867,6 +19210,18 @@ "type": "resref", "value": "thorn" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 32 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18039,6 +19394,18 @@ "type": "resref", "value": "dantarrow" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 11 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18242,6 +19609,18 @@ "type": "resref", "value": "isisbolt" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18279,7 +19658,7 @@ }, "Cost": { "type": "dword", - "value": 106400 + "value": 61608 }, "Cursed": { "type": "byte", @@ -18497,7 +19876,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -18507,6 +19886,18 @@ "type": "resref", "value": "mysticdagger" }, + "xModelPart1": { + "type": "word", + "value": 23 + }, + "xModelPart2": { + "type": "word", + "value": 33 + }, + "xModelPart3": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18989,6 +20380,18 @@ "type": "resref", "value": "dantgreatsword" }, + "xModelPart1": { + "type": "word", + "value": 23 + }, + "xModelPart2": { + "type": "word", + "value": 23 + }, + "xModelPart3": { + "type": "word", + "value": 23 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19347,6 +20750,18 @@ "type": "resref", "value": "steallongsword" }, + "xModelPart1": { + "type": "word", + "value": 33 + }, + "xModelPart2": { + "type": "word", + "value": 33 + }, + "xModelPart3": { + "type": "word", + "value": 11 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19384,7 +20799,7 @@ }, "Cost": { "type": "dword", - "value": 608996 + "value": 423048 }, "Cursed": { "type": "byte", @@ -19695,7 +21110,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -19705,6 +21120,18 @@ "type": "resref", "value": "stealkatana" }, + "xModelPart1": { + "type": "word", + "value": 41 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20032,6 +21459,18 @@ "type": "resref", "value": "dantrapier" }, + "xModelPart1": { + "type": "word", + "value": 43 + }, + "xModelPart2": { + "type": "word", + "value": 23 + }, + "xModelPart3": { + "type": "word", + "value": 43 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20452,6 +21891,18 @@ "type": "resref", "value": "dantmorningstar" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 33 + }, + "xModelPart3": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20934,6 +22385,18 @@ "type": "resref", "value": "danthalbred" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 11 + }, + "xModelPart3": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21168,6 +22631,18 @@ "type": "resref", "value": "isiscrossbow" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 43 + }, + "xModelPart3": { + "type": "word", + "value": 33 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21495,6 +22970,18 @@ "type": "resref", "value": "mysticbow" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 22 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21853,6 +23340,18 @@ "type": "resref", "value": "bowofbark" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, + "xModelPart2": { + "type": "word", + "value": 11 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22149,6 +23648,18 @@ "type": "resref", "value": "dantlongbow" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, + "xModelPart2": { + "type": "word", + "value": 33 + }, + "xModelPart3": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22438,7 +23949,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -22448,6 +23959,10 @@ "type": "resref", "value": "amuletofthewarri" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22726,7 +24241,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -22736,6 +24251,10 @@ "type": "resref", "value": "ringoflight" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23187,6 +24706,18 @@ "type": "resref", "value": "sorcstaffofpower" }, + "xModelPart1": { + "type": "word", + "value": 31 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 61 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23638,6 +25169,18 @@ "type": "resref", "value": "wizstaffofpower" }, + "xModelPart1": { + "type": "word", + "value": 32 + }, + "xModelPart2": { + "type": "word", + "value": 22 + }, + "xModelPart3": { + "type": "word", + "value": 62 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23779,6 +25322,10 @@ "type": "resref", "value": "banditstools" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 diff --git a/_module/git/hillsidemysticto.git.json b/_module/git/hillsidemysticto.git.json index eea2d12..4f8e2ae 100644 --- a/_module/git/hillsidemysticto.git.json +++ b/_module/git/hillsidemysticto.git.json @@ -411,7 +411,7 @@ }, "Cost": { "type": "dword", - "value": 2137445 + "value": 1473217 }, "Cursed": { "type": "byte", @@ -644,37 +644,6 @@ "value": 9 } }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 12 - }, - "Subtype": { - "type": "word", - "value": 15 - } - }, { "__struct_id": 0, "ChanceAppear": { @@ -954,37 +923,6 @@ "value": 2 } }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 37 - }, - "Subtype": { - "type": "word", - "value": 1 - } - }, { "__struct_id": 0, "ChanceAppear": { @@ -1179,16 +1117,92 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", - "value": "MysticRobe" + "value": "ISISMYSTICROBE" }, "TemplateResRef": { "type": "resref", "value": "mysticrobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 13 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 3 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1462,6 +1476,82 @@ "type": "resref", "value": "nw_cloth008" }, + "xArmorPart_Belt": { + "type": "word", + "value": 0 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 3 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 10 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 3 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 3 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 14 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 0 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1917,6 +2007,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 1 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": -1.0 @@ -2273,7 +2439,7 @@ }, "Cost": { "type": "dword", - "value": 2137445 + "value": 1473217 }, "Cursed": { "type": "byte", @@ -2506,37 +2672,6 @@ "value": 9 } }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 12 - }, - "Subtype": { - "type": "word", - "value": 15 - } - }, { "__struct_id": 0, "ChanceAppear": { @@ -2816,37 +2951,6 @@ "value": 2 } }, - { - "__struct_id": 0, - "ChanceAppear": { - "type": "byte", - "value": 100 - }, - "CostTable": { - "type": "byte", - "value": 0 - }, - "CostValue": { - "type": "word", - "value": 0 - }, - "Param1": { - "type": "byte", - "value": 255 - }, - "Param1Value": { - "type": "byte", - "value": 0 - }, - "PropertyName": { - "type": "word", - "value": 37 - }, - "Subtype": { - "type": "word", - "value": 1 - } - }, { "__struct_id": 0, "ChanceAppear": { @@ -3041,16 +3145,92 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", - "value": "MysticRobe" + "value": "ISISMYSTICROBE" }, "TemplateResRef": { "type": "resref", "value": "mysticrobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 13 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 1 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 3 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 10 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 7 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 8 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 8 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 19 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -3630,6 +3810,82 @@ "type": "byte", "value": 12 }, + "xAppearance_Head": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": -0.992479681968689 @@ -4003,6 +4259,10 @@ "type": "resref", "value": "awminofist" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4373,6 +4633,10 @@ "type": "resref", "value": "reaperbite" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -4742,6 +5006,10 @@ "type": "resref", "value": "spirtfist" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -5455,6 +5723,10 @@ "type": "resref", "value": "ruinsreaperprops" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -6898,6 +7170,18 @@ "type": "resref", "value": "nw_it_mpotion021" }, + "xModelPart1": { + "type": "word", + "value": 56 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7027,6 +7311,18 @@ "type": "resref", "value": "nw_it_mpotion021" }, + "xModelPart1": { + "type": "word", + "value": 56 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -7156,6 +7452,18 @@ "type": "resref", "value": "nw_it_mpotion021" }, + "xModelPart1": { + "type": "word", + "value": 56 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 21 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -16816,6 +17124,18 @@ "type": "resref", "value": "isisbolt" }, + "xModelPart1": { + "type": "word", + "value": 11 + }, + "xModelPart2": { + "type": "word", + "value": 21 + }, + "xModelPart3": { + "type": "word", + "value": 31 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17052,6 +17372,10 @@ "type": "resref", "value": "nw_it_sparscr603" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17280,6 +17604,10 @@ "type": "resref", "value": "x1_it_sparscr002" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17508,6 +17836,10 @@ "type": "resref", "value": "x1_it_sparscr102" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17736,6 +18068,10 @@ "type": "resref", "value": "nw_it_sparscr509" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -17902,6 +18238,10 @@ "type": "resref", "value": "x1_it_spdvscr204" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18068,6 +18408,10 @@ "type": "resref", "value": "x1_it_spdvscr701" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18296,6 +18640,10 @@ "type": "resref", "value": "x1_it_sparscr201" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18462,6 +18810,10 @@ "type": "resref", "value": "x1_it_spdvscr101" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18690,6 +19042,10 @@ "type": "resref", "value": "x1_it_spdvscr601" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -18918,6 +19274,10 @@ "type": "resref", "value": "nw_it_sparscr414" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19146,6 +19506,10 @@ "type": "resref", "value": "x1_it_sparscr801" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19374,6 +19738,10 @@ "type": "resref", "value": "x1_it_sparscr901" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19602,6 +19970,10 @@ "type": "resref", "value": "x1_it_sparscr602" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -19830,6 +20202,10 @@ "type": "resref", "value": "x1_it_sparscr701" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20058,6 +20434,10 @@ "type": "resref", "value": "x1_it_sparscr502" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20286,6 +20666,10 @@ "type": "resref", "value": "nw_it_sparscr211" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20452,6 +20836,10 @@ "type": "resref", "value": "x1_it_spdvscr202" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20618,6 +21006,10 @@ "type": "resref", "value": "x1_it_spdvscr803" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -20846,6 +21238,10 @@ "type": "resref", "value": "nw_it_sparscr212" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21074,6 +21470,10 @@ "type": "resref", "value": "nw_it_sparscr112" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21271,6 +21671,10 @@ "type": "resref", "value": "x1_it_spdvscr107" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21499,6 +21903,10 @@ "type": "resref", "value": "nw_it_sparscr213" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21727,6 +22135,10 @@ "type": "resref", "value": "nw_it_sparscr607" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -21955,6 +22367,10 @@ "type": "resref", "value": "nw_it_sparscr405" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22183,6 +22599,10 @@ "type": "resref", "value": "nw_it_sparscr107" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22411,6 +22831,10 @@ "type": "resref", "value": "nw_it_spdvscr202" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22639,6 +23063,10 @@ "type": "resref", "value": "nw_it_sparscr610" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -22867,6 +23295,10 @@ "type": "resref", "value": "nw_it_sparscr307" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23095,6 +23527,10 @@ "type": "resref", "value": "nw_it_sparscr217" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23323,6 +23759,10 @@ "type": "resref", "value": "nw_it_sparscr502" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23551,6 +23991,10 @@ "type": "resref", "value": "nw_it_sparscr110" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -23779,6 +24223,10 @@ "type": "resref", "value": "nw_it_sparscr507" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24007,6 +24455,10 @@ "type": "resref", "value": "nw_it_sparscr406" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24235,6 +24687,10 @@ "type": "resref", "value": "nw_it_sparscr411" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24463,6 +24919,10 @@ "type": "resref", "value": "x1_it_spdvscr301" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24691,6 +25151,10 @@ "type": "resref", "value": "nw_it_sparscr707" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -24857,6 +25321,10 @@ "type": "resref", "value": "x1_it_spdvscr804" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25085,6 +25553,10 @@ "type": "resref", "value": "x1_it_spdvscr605" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25251,6 +25723,10 @@ "type": "resref", "value": "x1_it_spdvscr702" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25479,6 +25955,10 @@ "type": "resref", "value": "nw_it_sparscr206" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25707,6 +26187,10 @@ "type": "resref", "value": "nw_it_sparscr003" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -25935,6 +26419,10 @@ "type": "resref", "value": "nw_it_sparscr704" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26101,6 +26589,10 @@ "type": "resref", "value": "x1_it_spdvscr703" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26329,6 +26821,10 @@ "type": "resref", "value": "x1_it_sparscr601" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26557,6 +27053,10 @@ "type": "resref", "value": "nw_it_sparscr501" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -26785,6 +27285,10 @@ "type": "resref", "value": "nw_it_sparscr301" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27013,6 +27517,10 @@ "type": "resref", "value": "x1_it_sparscr301" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27210,6 +27718,10 @@ "type": "resref", "value": "x1_it_spdvscr102" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27438,6 +27950,10 @@ "type": "resref", "value": "nw_it_sparscr905" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27666,6 +28182,10 @@ "type": "resref", "value": "nw_it_sparscr503" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -27832,6 +28352,10 @@ "type": "resref", "value": "x1_it_spdvscr604" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28060,6 +28584,10 @@ "type": "resref", "value": "nw_it_sparscr219" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28257,6 +28785,10 @@ "type": "resref", "value": "x1_it_spdvscr801" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28485,6 +29017,10 @@ "type": "resref", "value": "x1_it_sparscr003" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28713,6 +29249,10 @@ "type": "resref", "value": "nw_it_sparscr416" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -28941,6 +29481,10 @@ "type": "resref", "value": "nw_it_sparscr215" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29169,6 +29713,10 @@ "type": "resref", "value": "nw_it_sparscr101" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29397,6 +29945,10 @@ "type": "resref", "value": "nw_it_sparscr908" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29625,6 +30177,10 @@ "type": "resref", "value": "nw_it_sparscr412" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -29791,6 +30347,10 @@ "type": "resref", "value": "x1_it_spdvscr103" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30019,6 +30579,10 @@ "type": "resref", "value": "nw_it_sparscr608" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30247,6 +30811,10 @@ "type": "resref", "value": "nw_it_sparscr418" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30475,6 +31043,10 @@ "type": "resref", "value": "x1_it_sparscr101" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30703,6 +31275,10 @@ "type": "resref", "value": "nw_it_sparscr413" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -30931,6 +31507,10 @@ "type": "resref", "value": "nw_it_sparscr504" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31159,6 +31739,10 @@ "type": "resref", "value": "nw_it_sparscr708" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31356,6 +31940,10 @@ "type": "resref", "value": "x1_it_spdvscr704" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31584,6 +32172,10 @@ "type": "resref", "value": "nw_it_sparscr309" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -31812,6 +32404,10 @@ "type": "resref", "value": "x1_it_sparscr501" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32040,6 +32636,10 @@ "type": "resref", "value": "nw_it_sparscr304" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32206,6 +32806,10 @@ "type": "resref", "value": "x1_it_spdvscr205" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32403,6 +33007,10 @@ "type": "resref", "value": "x1_it_spdvscr403" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32631,6 +33239,10 @@ "type": "resref", "value": "x1_it_sparscr001" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -32859,6 +33471,10 @@ "type": "resref", "value": "x1_it_sparscr605" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33087,6 +33703,10 @@ "type": "resref", "value": "nw_it_sparscr220" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33315,6 +33935,10 @@ "type": "resref", "value": "nw_it_sparscr902" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33543,6 +34167,10 @@ "type": "resref", "value": "nw_it_sparscr208" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33771,6 +34399,10 @@ "type": "resref", "value": "nw_it_sparscr209" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -33999,6 +34631,10 @@ "type": "resref", "value": "nw_it_sparscr601" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34227,6 +34863,10 @@ "type": "resref", "value": "nw_it_sparscr103" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34455,6 +35095,10 @@ "type": "resref", "value": "nw_it_sparscr602" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34652,6 +35296,10 @@ "type": "resref", "value": "x1_it_spdvscr303" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -34880,6 +35528,10 @@ "type": "resref", "value": "nw_it_sparscr803" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35015,6 +35667,10 @@ "type": "resref", "value": "nw_it_spdvscr701" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35243,6 +35899,10 @@ "type": "resref", "value": "x1_it_spdvscr602" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35471,6 +36131,10 @@ "type": "resref", "value": "nw_it_sparscr508" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35699,6 +36363,10 @@ "type": "resref", "value": "nw_it_sparscr612" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -35927,6 +36595,10 @@ "type": "resref", "value": "nw_it_sparscr912" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36155,6 +36827,10 @@ "type": "resref", "value": "nw_it_sparscr613" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36383,6 +37059,10 @@ "type": "resref", "value": "x1_it_sparscr303" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36611,6 +37291,10 @@ "type": "resref", "value": "nw_it_sparscr312" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -36839,6 +37523,10 @@ "type": "resref", "value": "nw_it_sparscr505" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37067,6 +37755,10 @@ "type": "resref", "value": "nw_it_sparscr308" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37295,6 +37987,10 @@ "type": "resref", "value": "nw_it_sparscr809" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37523,6 +38219,10 @@ "type": "resref", "value": "nw_it_sparscr106" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37751,6 +38451,10 @@ "type": "resref", "value": "nw_it_sparscr804" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -37917,6 +38621,10 @@ "type": "resref", "value": "x1_it_spdvscr501" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38083,6 +38791,10 @@ "type": "resref", "value": "x1_it_spdvscr401" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38249,6 +38961,10 @@ "type": "resref", "value": "x1_it_spdvscr104" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38415,6 +39131,10 @@ "type": "resref", "value": "x1_it_spdvscr001" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38581,6 +39301,10 @@ "type": "resref", "value": "x1_it_spdvscr201" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38747,6 +39471,10 @@ "type": "resref", "value": "x1_it_spdvscr302" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -38975,6 +39703,10 @@ "type": "resref", "value": "x1_it_sparscr603" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39203,6 +39935,10 @@ "type": "resref", "value": "x1_it_sparscr401" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39431,6 +40167,10 @@ "type": "resref", "value": "nw_it_sparscr216" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39659,6 +40399,10 @@ "type": "resref", "value": "nw_it_sparscr218" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -39887,6 +40631,10 @@ "type": "resref", "value": "nw_it_sparscr511" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40115,6 +40863,10 @@ "type": "resref", "value": "nw_it_sparscr512" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40250,6 +41002,10 @@ "type": "resref", "value": "nw_it_spdvscr201" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40478,6 +41234,10 @@ "type": "resref", "value": "nw_it_sparscr417" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40706,6 +41466,10 @@ "type": "resref", "value": "nw_it_sparscr513" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -40934,6 +41698,10 @@ "type": "resref", "value": "nw_it_sparscr004" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41162,6 +41930,10 @@ "type": "resref", "value": "nw_it_sparscr310" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41390,6 +42162,10 @@ "type": "resref", "value": "nw_it_sparscr104" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41618,6 +42394,10 @@ "type": "resref", "value": "nw_it_sparscr302" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -41815,6 +42595,10 @@ "type": "resref", "value": "x1_it_spdvscr106" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42043,6 +42827,10 @@ "type": "resref", "value": "nw_it_sparscr109" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42271,6 +43059,10 @@ "type": "resref", "value": "nw_it_sparscr807" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42468,6 +43260,10 @@ "type": "resref", "value": "x1_it_spdvscr402" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42696,6 +43492,10 @@ "type": "resref", "value": "nw_it_sparscr806" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -42924,6 +43724,10 @@ "type": "resref", "value": "nw_it_sparscr611" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -43152,6 +43956,10 @@ "type": "resref", "value": "nw_it_sparscr202" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -43380,6 +44188,10 @@ "type": "resref", "value": "nw_it_sparscr906" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -43608,6 +44420,10 @@ "type": "resref", "value": "nw_it_sparscr801" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -43836,6 +44652,10 @@ "type": "resref", "value": "nw_it_sparscr506" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -44064,6 +44884,10 @@ "type": "resref", "value": "nw_it_sparscr401" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -44292,6 +45116,10 @@ "type": "resref", "value": "nw_it_sparscr901" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -44520,6 +45348,10 @@ "type": "resref", "value": "nw_it_sparscr705" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -44748,6 +45580,10 @@ "type": "resref", "value": "nw_it_sparscr315" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -44976,6 +45812,10 @@ "type": "resref", "value": "nw_it_sparscr113" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -45111,6 +45951,10 @@ "type": "resref", "value": "nw_it_spdvscr402" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -45308,6 +46152,10 @@ "type": "resref", "value": "x1_it_spdvscr203" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -45474,6 +46322,10 @@ "type": "resref", "value": "x1_it_spdvscr502" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -45702,6 +46554,10 @@ "type": "resref", "value": "nw_it_sparscr221" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -45930,6 +46786,10 @@ "type": "resref", "value": "nw_it_sparscr409" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -46096,6 +46956,10 @@ "type": "resref", "value": "x1_it_spdvscr603" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -46324,6 +47188,10 @@ "type": "resref", "value": "nw_it_sparscr604" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -46552,6 +47420,10 @@ "type": "resref", "value": "nw_it_sparscr415" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -46780,6 +47652,10 @@ "type": "resref", "value": "nw_it_sparscr903" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -47008,6 +47884,10 @@ "type": "resref", "value": "nw_it_sparscr702" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -47236,6 +48116,10 @@ "type": "resref", "value": "nw_it_sparscr808" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -47464,6 +48348,10 @@ "type": "resref", "value": "nw_it_sparscr706" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -47754,6 +48642,10 @@ "type": "resref", "value": "nw_it_sparscr102" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -47982,6 +48874,10 @@ "type": "resref", "value": "nw_it_sparscr303" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -48210,6 +49106,10 @@ "type": "resref", "value": "nw_it_sparscr802" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -48376,6 +49276,10 @@ "type": "resref", "value": "x1_it_spdvscr305" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -48511,6 +49415,10 @@ "type": "resref", "value": "nw_it_spdvscr501" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -48739,6 +49647,10 @@ "type": "resref", "value": "nw_it_sparscr111" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -48967,6 +49879,10 @@ "type": "resref", "value": "nw_it_sparscr002" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -49102,6 +50018,10 @@ "type": "resref", "value": "nw_it_spdvscr301" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -49330,6 +50250,10 @@ "type": "resref", "value": "nw_it_sparscr402" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -49465,6 +50389,10 @@ "type": "resref", "value": "nw_it_spdvscr302" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -49693,6 +50621,10 @@ "type": "resref", "value": "nw_it_sparscr201" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -49921,6 +50853,10 @@ "type": "resref", "value": "nw_it_sparscr001" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -50056,6 +50992,10 @@ "type": "resref", "value": "nw_it_spdvscr401" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -50191,6 +51131,10 @@ "type": "resref", "value": "nw_it_spdvscr702" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -50419,6 +51363,10 @@ "type": "resref", "value": "nw_it_sparscr210" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -50647,6 +51595,10 @@ "type": "resref", "value": "nw_it_sparscr205" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -50875,6 +51827,10 @@ "type": "resref", "value": "nw_it_sparscr609" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -51103,6 +52059,10 @@ "type": "resref", "value": "nw_it_sparscr410" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -51331,6 +52291,10 @@ "type": "resref", "value": "nw_it_sparscr910" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -51559,6 +52523,10 @@ "type": "resref", "value": "x1_it_sparscr103" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -51725,6 +52693,10 @@ "type": "resref", "value": "x1_it_spdvscr105" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -51953,6 +52925,10 @@ "type": "resref", "value": "nw_it_spdvscr203" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -52181,6 +53157,10 @@ "type": "resref", "value": "nw_it_sparscr108" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -52409,6 +53389,10 @@ "type": "resref", "value": "nw_it_sparscr313" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -52637,6 +53621,10 @@ "type": "resref", "value": "nw_it_spdvscr204" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -52865,6 +53853,10 @@ "type": "resref", "value": "nw_it_sparscr701" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -53031,6 +54023,10 @@ "type": "resref", "value": "x1_it_spdvscr304" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -53259,6 +54255,10 @@ "type": "resref", "value": "nw_it_sparscr305" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -53487,6 +54487,10 @@ "type": "resref", "value": "x1_it_sparscr604" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -53715,6 +54719,10 @@ "type": "resref", "value": "nw_it_sparscr403" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -53943,6 +54951,10 @@ "type": "resref", "value": "nw_it_sparscr105" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -54171,6 +55183,10 @@ "type": "resref", "value": "nw_it_sparscr203" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -54399,6 +55415,10 @@ "type": "resref", "value": "nw_it_sparscr306" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -54627,6 +55647,10 @@ "type": "resref", "value": "nw_it_sparscr404" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -54855,6 +55879,10 @@ "type": "resref", "value": "nw_it_sparscr904" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -55083,6 +56111,10 @@ "type": "resref", "value": "nw_it_sparscr510" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -55311,6 +56343,10 @@ "type": "resref", "value": "nw_it_sparscr605" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -55539,6 +56575,10 @@ "type": "resref", "value": "nw_it_sparscr703" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -55767,6 +56807,10 @@ "type": "resref", "value": "nw_it_sparscr805" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -55995,6 +57039,10 @@ "type": "resref", "value": "x1_it_spdvscr802" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -56223,6 +57271,10 @@ "type": "resref", "value": "x1_it_sparscr202" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -56451,6 +57503,10 @@ "type": "resref", "value": "nw_it_sparscr614" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -56679,6 +57735,10 @@ "type": "resref", "value": "nw_it_sparscr911" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -56907,6 +57967,10 @@ "type": "resref", "value": "nw_it_sparscr606" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -57135,6 +58199,10 @@ "type": "resref", "value": "x1_it_sparscr104" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -57363,6 +58431,10 @@ "type": "resref", "value": "nw_it_sparscr214" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -57529,6 +58601,10 @@ "type": "resref", "value": "x1_it_spdvscr901" }, + "xModelPart1": { + "type": "word", + "value": 0 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -57757,6 +58833,10 @@ "type": "resref", "value": "nw_it_sparscr311" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -57985,6 +59065,10 @@ "type": "resref", "value": "nw_it_sparscr909" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -58213,6 +59297,10 @@ "type": "resref", "value": "nw_it_sparscr407" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -58441,6 +59529,10 @@ "type": "resref", "value": "nw_it_sparscr204" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -58669,6 +59761,10 @@ "type": "resref", "value": "nw_it_sparscr907" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -58893,6 +59989,10 @@ "type": "resref", "value": "nw_it_sparscr603" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -58938,7 +60038,7 @@ }, "Cost": { "type": "dword", - "value": 131970 + "value": 351751 }, "Cursed": { "type": "byte", @@ -59113,7 +60213,7 @@ }, "CostValue": { "type": "word", - "value": 2 + "value": 4 }, "Param1": { "type": "byte", @@ -59144,7 +60244,7 @@ }, "CostValue": { "type": "word", - "value": 2 + "value": 4 }, "Param1": { "type": "byte", @@ -59179,7 +60279,7 @@ }, "Stolen": { "type": "byte", - "value": 1 + "value": 0 }, "Tag": { "type": "cexostring", @@ -59189,6 +60289,10 @@ "type": "resref", "value": "ringoflight" }, + "xModelPart1": { + "type": "word", + "value": 13 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -59353,6 +60457,10 @@ "type": "resref", "value": "item" }, + "xModelPart1": { + "type": "word", + "value": 2 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -59718,6 +60826,18 @@ "type": "resref", "value": "thorn" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, + "xModelPart2": { + "type": "word", + "value": 32 + }, + "xModelPart3": { + "type": "word", + "value": 22 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -59933,6 +61053,10 @@ "type": "resref", "value": "hillsiflameberry" }, + "xModelPart1": { + "type": "word", + "value": 66 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -60134,6 +61258,10 @@ "type": "resref", "value": "hillsideiceberry" }, + "xModelPart1": { + "type": "word", + "value": 66 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -60269,6 +61397,10 @@ "type": "resref", "value": "nw_it_msmlmisc23" }, + "xModelPart1": { + "type": "word", + "value": 12 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -60404,6 +61536,10 @@ "type": "resref", "value": "nw_it_msmlmisc24" }, + "xModelPart1": { + "type": "word", + "value": 67 + }, "XOrientation": { "type": "float", "value": 0.0 diff --git a/_module/git/ifm_pc_rooms.git.json b/_module/git/ifm_pc_rooms.git.json new file mode 100644 index 0000000..8c0af71 --- /dev/null +++ b/_module/git/ifm_pc_rooms.git.json @@ -0,0 +1,2582 @@ +{ + "__data_type": "GIT ", + "AreaProperties": { + "__struct_id": 100, + "type": "struct", + "value": { + "__struct_id": 100, + "AmbientSndDay": { + "type": "int", + "value": 72 + }, + "AmbientSndDayVol": { + "type": "int", + "value": 32 + }, + "AmbientSndNight": { + "type": "int", + "value": 107 + }, + "AmbientSndNitVol": { + "type": "int", + "value": 12 + }, + "EnvAudio": { + "type": "int", + "value": 2 + }, + "MusicBattle": { + "type": "int", + "value": 40 + }, + "MusicDay": { + "type": "int", + "value": 25 + }, + "MusicDelay": { + "type": "int", + "value": 45000 + }, + "MusicNight": { + "type": "int", + "value": 25 + } + } + }, + "Creature List": { + "type": "list", + "value": [] + }, + "Door List": { + "type": "list", + "value": [ + { + "__struct_id": 8, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 0 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "The only exit from your room at the Flying Monkey.", + "id": 9078 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "GenericType_New": { + "type": "dword", + "value": 12 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "LinkedTo": { + "type": "cexostring", + "value": "pcrooms2inn" + }, + "LinkedToFlags": { + "type": "byte", + "value": 2 + }, + "LoadScreenID": { + "type": "word", + "value": 42 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Door", + "id": 5349 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "se_door_death" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnFailToOpen": { + "type": "resref", + "value": "door_fail2open" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "ra_close_door" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 0 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "pcrooms2inn" + }, + "TemplateResRef": { + "type": "resref", + "value": "x3_door_wood002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 15.0 + }, + "Y": { + "type": "float", + "value": 20.0 + }, + "Z": { + "type": "float", + "value": 0.0 + } + } + ] + }, + "Encounter List": { + "type": "list", + "value": [] + }, + "List": { + "type": "list", + "value": [] + }, + "Placeable List": { + "type": "list", + "value": [ + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2802 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.7853981852531433 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16813643 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16811120 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 0 + }, + "PortraitId": { + "type": "word", + "value": 408 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_BED002" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_bed002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 16.78469085693359 + }, + "Y": { + "type": "float", + "value": 13.20924949645996 + }, + "Z": { + "type": "float", + "value": 0.009990006685256958 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 0 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.570796251296997 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "This looks like a safe place to store your gear... for now.", + "id": 14548 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Armoire", + "id": 14549 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "obj_us_chest" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 358 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "Armoire" + }, + "TemplateResRef": { + "type": "resref", + "value": "plc_armoire" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.55000019073486 + }, + "Y": { + "type": "float", + "value": 16.82999992370606 + }, + "Z": { + "type": "float", + "value": 0.0 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2207 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.8590291738510132 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "You see yourself in the mirror.", + "id": 16810932 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Mirror", + "id": 16812334 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "sob_examine002" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 2558 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_MIRROR005" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_mirror005" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.54697418212891 + }, + "Y": { + "type": "float", + "value": 18.48270606994629 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 12 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": -1.570796251296997 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 14607 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 0 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 5655 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "nw_o2_generallow" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "nw_o2_generallow" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 370 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "Desk" + }, + "TemplateResRef": { + "type": "resref", + "value": "plc_desk" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 18.48391151428223 + }, + "Y": { + "type": "float", + "value": 17.2309455871582 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 179 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.006291270256043 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "It is a simple chair but the grace of its lines speaks to the quality of its craftmanship.", + "id": 14593 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Chair", + "id": 5822 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "sei_sit" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 537 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "Chair" + }, + "TemplateResRef": { + "type": "resref", + "value": "chair001" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 17.41463851928711 + }, + "Y": { + "type": "float", + "value": 17.40582275390625 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2813 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 2.012582778930664 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16813707 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16813470 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 2564 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_TUB002" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_tub002" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.75813961029053 + }, + "Y": { + "type": "float", + "value": 12.54897880554199 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2298 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.0 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16810931 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16813468 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 2563 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_STONEBSIN003" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_stonebsin003" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 11.57714176177979 + }, + "Y": { + "type": "float", + "value": 11.53489303588867 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2976 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.3681553304195404 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 2 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16813656 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 2 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16812700 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 2500 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_SCREEN014" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_screen014" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 12.03563499450684 + }, + "Y": { + "type": "float", + "value": 13.57882213592529 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 1914 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": -3.141592264175415 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "A simple throw rug adorns the floor, faded and worn with use.", + "id": 16813674 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16813108 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 533 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "PC_ROOM_MAT" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_throwrug058" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 14.97459030151367 + }, + "Y": { + "type": "float", + "value": 18.7603874206543 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 2565 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 0.0 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 10 + }, + "Description": { + "type": "cexolocstring", + "value": { + "id": 16810853 + } + }, + "DisarmDC": { + "type": "byte", + "value": 0 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 5 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 10 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 1 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "id": 16811458 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 0 + }, + "PortraitId": { + "type": "word", + "value": 510 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 1 + }, + "Tag": { + "type": "cexostring", + "value": "ZEP_CITYINTRIORF" + }, + "TemplateResRef": { + "type": "resref", + "value": "zep_cityintriorf" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 14.61999988555908 + }, + "Y": { + "type": "float", + "value": 14.60000038146973 + }, + "Z": { + "type": "float", + "value": 0.009999999776482582 + } + } + ] + }, + "SoundList": { + "type": "list", + "value": [] + }, + "StoreList": { + "type": "list", + "value": [] + }, + "TriggerList": { + "type": "list", + "value": [] + }, + "WaypointList": { + "type": "list", + "value": [] + } +} diff --git a/_module/git/trinitybank.git.json b/_module/git/trinitybank.git.json index a3c12f4..94609d4 100644 --- a/_module/git/trinitybank.git.json +++ b/_module/git/trinitybank.git.json @@ -438,6 +438,82 @@ "type": "resref", "value": "bankeroutfit" }, + "xArmorPart_Belt": { + "type": "word", + "value": 4 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 4 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 10 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 3 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -559,6 +635,10 @@ "type": "resref", "value": "nw_it_crewpsp001" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -680,6 +760,10 @@ "type": "resref", "value": "nw_it_crewpsp001" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -801,6 +885,10 @@ "type": "resref", "value": "nw_it_crewps001" }, + "xModelPart1": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1092,6 +1180,82 @@ "type": "resref", "value": "newplayerhelperr" }, + "xArmorPart_Belt": { + "type": "word", + "value": 4 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 4 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 4 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 4 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 4 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1348,6 +1512,82 @@ "type": "resref", "value": "noblerobe" }, + "xArmorPart_Belt": { + "type": "word", + "value": 4 + }, + "xArmorPart_LBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_LFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_LHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_LShou": { + "type": "word", + "value": 4 + }, + "xArmorPart_LThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Neck": { + "type": "word", + "value": 4 + }, + "xArmorPart_Pelvi": { + "type": "word", + "value": 4 + }, + "xArmorPart_RBice": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFArm": { + "type": "word", + "value": 4 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 4 + }, + "xArmorPart_RHand": { + "type": "word", + "value": 4 + }, + "xArmorPart_Robe": { + "type": "word", + "value": 0 + }, + "xArmorPart_RShin": { + "type": "word", + "value": 4 + }, + "xArmorPart_RShou": { + "type": "word", + "value": 4 + }, + "xArmorPart_RThig": { + "type": "word", + "value": 4 + }, + "xArmorPart_Torso": { + "type": "word", + "value": 53 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -1803,6 +2043,82 @@ "type": "byte", "value": 10 }, + "xAppearance_Head": { + "type": "word", + "value": 3 + }, + "xArmorPart_RFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_Belt": { + "type": "word", + "value": 0 + }, + "xBodyPart_LBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_LFoot": { + "type": "word", + "value": 1 + }, + "xBodyPart_LHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_LShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_LThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Neck": { + "type": "word", + "value": 1 + }, + "xBodyPart_Pelvis": { + "type": "word", + "value": 1 + }, + "xBodyPart_RBicep": { + "type": "word", + "value": 1 + }, + "xBodyPart_RFArm": { + "type": "word", + "value": 1 + }, + "xBodyPart_RHand": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShin": { + "type": "word", + "value": 1 + }, + "xBodyPart_RShoul": { + "type": "word", + "value": 0 + }, + "xBodyPart_RThigh": { + "type": "word", + "value": 1 + }, + "xBodyPart_Torso": { + "type": "word", + "value": 1 + }, "XOrientation": { "type": "float", "value": 0.0 @@ -8016,6 +8332,437 @@ "type": "float", "value": 0.0 } + }, + { + "__struct_id": 9, + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 445 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "Bearing": { + "type": "float", + "value": 1.570796251296997 + }, + "BodyBag": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Conversation": { + "type": "resref", + "value": "" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "You can safely* store all of your gear here.\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n* actualy saftey may not apply", + "id": 84184 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HasInventory": { + "type": "byte", + "value": 0 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 0 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "Arcane Deposit Box", + "id": 83439 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnInvDisturbed": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUsed": { + "type": "resref", + "value": "nui_f_storage" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 928 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Static": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "ARCANE_STORAGE" + }, + "TemplateResRef": { + "type": "resref", + "value": "x2_plc_chest_drw" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Type": { + "type": "byte", + "value": 0 + }, + "Useable": { + "type": "byte", + "value": 1 + }, + "VisTransformList": { + "type": "list", + "value": [ + { + "__struct_id": 6, + "AnimationSpeed": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 1.0 + } + } + }, + "RotateX": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + }, + "RotateY": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + }, + "RotateZ": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + }, + "ScaleX": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 1.551327705383301 + } + } + }, + "ScaleY": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 1.551327705383301 + } + } + }, + "ScaleZ": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 1.551327705383301 + } + } + }, + "Scope": { + "type": "int", + "value": 0 + }, + "TranslateX": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + }, + "TranslateY": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + }, + "TranslateZ": { + "__struct_id": 0, + "type": "struct", + "value": { + "__struct_id": 0, + "LerpType": { + "type": "int", + "value": 0 + }, + "TimerType": { + "type": "int", + "value": 0 + }, + "ValueTo": { + "type": "float", + "value": 0.0 + } + } + } + } + ] + }, + "Will": { + "type": "byte", + "value": 0 + }, + "X": { + "type": "float", + "value": 22.39899444580078 + }, + "Y": { + "type": "float", + "value": 8.892400741577148 + }, + "Z": { + "type": "float", + "value": -5.7220458984375e-006 + } } ] }, diff --git a/_module/ifo/module.ifo.json b/_module/ifo/module.ifo.json index 2d40e16..52b4229 100644 --- a/_module/ifo/module.ifo.json +++ b/_module/ifo/module.ifo.json @@ -2456,6 +2456,20 @@ "type": "resref", "value": "wol_a_dymond" } + }, + { + "__struct_id": 6, + "Area_Name": { + "type": "resref", + "value": "ifm_pc_rooms" + } + }, + { + "__struct_id": 6, + "Area_Name": { + "type": "resref", + "value": "dbi_pc_rooms" + } } ] }, @@ -2652,7 +2666,7 @@ }, "Mod_OnNuiEvent": { "type": "resref", - "value": "" + "value": "mod_nuievent" }, "Mod_OnPlrChat": { "type": "resref", @@ -2684,7 +2698,7 @@ }, "Mod_OnPlrTarget": { "type": "resref", - "value": "prc_onplaytarget" + "value": "mod_onplaytarget" }, "Mod_OnPlrTileAct": { "type": "resref", diff --git a/_module/itp/creaturepalcus.itp.json b/_module/itp/creaturepalcus.itp.json index 0c29773..1d3e4d5 100644 --- a/_module/itp/creaturepalcus.itp.json +++ b/_module/itp/creaturepalcus.itp.json @@ -4052,6 +4052,25 @@ "LIST": { "type": "list", "value": [ + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 0.3333333432674408 + }, + "FACTION": { + "type": "cexostring", + "value": "Hostile" + }, + "NAME": { + "type": "cexostring", + "value": "Hobgoblin Thrall" + }, + "RESREF": { + "type": "resref", + "value": "prc_wrsl_war" + } + }, { "__struct_id": 0, "CR": { @@ -5220,6 +5239,25 @@ "LIST": { "type": "list", "value": [ + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 7.0 + }, + "FACTION": { + "type": "cexostring", + "value": "Hostile" + }, + "NAME": { + "type": "cexostring", + "value": "Algoid" + }, + "RESREF": { + "type": "resref", + "value": "prc_algoid" + } + }, { "__struct_id": 0, "CR": { @@ -5258,6 +5296,44 @@ "value": "mastercain001" } }, + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 1.0 + }, + "FACTION": { + "type": "cexostring", + "value": "Commoner" + }, + "NAME": { + "type": "cexostring", + "value": "Myconid" + }, + "RESREF": { + "type": "resref", + "value": "prc_myconid" + } + }, + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 6.0 + }, + "FACTION": { + "type": "cexostring", + "value": "Hostile" + }, + "NAME": { + "type": "cexostring", + "value": "Shambling Mound" + }, + "RESREF": { + "type": "resref", + "value": "prc_shambmound" + } + }, { "__struct_id": 0, "CR": { @@ -13063,6 +13139,25 @@ "value": "hamatula" } }, + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 8.0 + }, + "FACTION": { + "type": "cexostring", + "value": "Commoner" + }, + "NAME": { + "type": "cexostring", + "value": "Hamatula" + }, + "RESREF": { + "type": "resref", + "value": "hamatula001" + } + }, { "__struct_id": 0, "CR": { @@ -14051,6 +14146,25 @@ "value": "prc_pm_mage2" } }, + { + "__struct_id": 0, + "CR": { + "type": "float", + "value": 3.0 + }, + "FACTION": { + "type": "cexostring", + "value": "Defender" + }, + "NAME": { + "type": "cexostring", + "value": "Spiritual Weapon" + }, + "RESREF": { + "type": "resref", + "value": "prc_spirit_weapn" + } + }, { "__struct_id": 0, "CR": { diff --git a/_module/itp/doorpalcus.itp.json b/_module/itp/doorpalcus.itp.json index ad3efde..ac084e3 100644 --- a/_module/itp/doorpalcus.itp.json +++ b/_module/itp/doorpalcus.itp.json @@ -17,6 +17,17 @@ "LIST": { "type": "list", "value": [ + { + "__struct_id": 0, + "NAME": { + "type": "cexostring", + "value": "\"Create Area\" Door" + }, + "RESREF": { + "type": "resref", + "value": "inn2pcrooms" + } + }, { "__struct_id": 0, "NAME": { diff --git a/_module/itp/placeablepalcus.itp.json b/_module/itp/placeablepalcus.itp.json index eed38c1..4f0cef3 100644 --- a/_module/itp/placeablepalcus.itp.json +++ b/_module/itp/placeablepalcus.itp.json @@ -1770,7 +1770,7 @@ "__struct_id": 0, "NAME": { "type": "cexostring", - "value": "Genesis Enterance Portal" + "value": "Genesis Entrance Portal" }, "RESREF": { "type": "resref", diff --git a/_module/jrl/module.jrl.json b/_module/jrl/module.jrl.json index d90ec2e..ee038ad 100644 --- a/_module/jrl/module.jrl.json +++ b/_module/jrl/module.jrl.json @@ -56,6 +56,57 @@ }, { "__struct_id": 1, + "Comment": { + "type": "cexostring", + "value": "" + }, + "EntryList": { + "type": "list", + "value": [ + { + "__struct_id": 0, + "End": { + "type": "word", + "value": 0 + }, + "ID": { + "type": "dword", + "value": 1 + }, + "Text": { + "type": "cexolocstring", + "value": { + "0": "Visit the PRC8 Discord for ruleset information, as well as other PW servers & dozens of single player PRC8 modules.\n\nhttps://discord.gg/FW9V9RKy5U" + } + } + } + ] + }, + "Name": { + "type": "cexolocstring", + "value": { + "0": "| PRC8 Discord |" + } + }, + "Picture": { + "type": "word", + "value": 65535 + }, + "Priority": { + "type": "dword", + "value": 4 + }, + "Tag": { + "type": "cexostring", + "value": "JRNL_PRC8" + }, + "XP": { + "type": "dword", + "value": 0 + } + }, + { + "__struct_id": 2, "Comment": { "type": "cexostring", "value": "3e D&D Experience Chart" @@ -106,7 +157,7 @@ } }, { - "__struct_id": 2, + "__struct_id": 3, "Comment": { "type": "cexostring", "value": "" @@ -174,7 +225,7 @@ } }, { - "__struct_id": 3, + "__struct_id": 4, "Comment": { "type": "cexostring", "value": "" @@ -242,7 +293,7 @@ } }, { - "__struct_id": 4, + "__struct_id": 5, "Comment": { "type": "cexostring", "value": "" @@ -310,7 +361,7 @@ } }, { - "__struct_id": 5, + "__struct_id": 6, "Comment": { "type": "cexostring", "value": "" @@ -378,7 +429,7 @@ } }, { - "__struct_id": 6, + "__struct_id": 7, "Comment": { "type": "cexostring", "value": "" @@ -480,7 +531,7 @@ } }, { - "__struct_id": 7, + "__struct_id": 8, "Comment": { "type": "cexostring", "value": "" @@ -548,7 +599,7 @@ } }, { - "__struct_id": 8, + "__struct_id": 9, "Comment": { "type": "cexostring", "value": "" @@ -633,7 +684,7 @@ } }, { - "__struct_id": 9, + "__struct_id": 10, "Comment": { "type": "cexostring", "value": "" @@ -769,7 +820,7 @@ } }, { - "__struct_id": 10, + "__struct_id": 11, "Comment": { "type": "cexostring", "value": "" @@ -837,7 +888,7 @@ } }, { - "__struct_id": 11, + "__struct_id": 12, "Comment": { "type": "cexostring", "value": "" @@ -905,7 +956,7 @@ } }, { - "__struct_id": 12, + "__struct_id": 13, "Comment": { "type": "cexostring", "value": "" @@ -973,7 +1024,7 @@ } }, { - "__struct_id": 13, + "__struct_id": 14, "Comment": { "type": "cexostring", "value": "" @@ -1075,7 +1126,7 @@ } }, { - "__struct_id": 14, + "__struct_id": 15, "Comment": { "type": "cexostring", "value": "" @@ -1177,7 +1228,7 @@ } }, { - "__struct_id": 15, + "__struct_id": 16, "Comment": { "type": "cexostring", "value": "" @@ -1245,7 +1296,7 @@ } }, { - "__struct_id": 16, + "__struct_id": 17, "Comment": { "type": "cexostring", "value": "" @@ -1296,7 +1347,7 @@ } }, { - "__struct_id": 17, + "__struct_id": 18, "Comment": { "type": "cexostring", "value": "" diff --git a/_module/ncs/0_chair_sit.ncs b/_module/ncs/0_chair_sit.ncs index 607923d..97b7996 100644 Binary files a/_module/ncs/0_chair_sit.ncs and b/_module/ncs/0_chair_sit.ncs differ diff --git a/_module/ncs/0_conv_placable.ncs b/_module/ncs/0_conv_placable.ncs index 3a77cba..864648d 100644 Binary files a/_module/ncs/0_conv_placable.ncs and b/_module/ncs/0_conv_placable.ncs differ diff --git a/_module/ncs/0_disarmsuccess.ncs b/_module/ncs/0_disarmsuccess.ncs index e7ee794..2ddadb0 100644 Binary files a/_module/ncs/0_disarmsuccess.ncs and b/_module/ncs/0_disarmsuccess.ncs differ diff --git a/_module/ncs/0_door_limit.ncs b/_module/ncs/0_door_limit.ncs index 2649654..0e4599a 100644 Binary files a/_module/ncs/0_door_limit.ncs and b/_module/ncs/0_door_limit.ncs differ diff --git a/_module/ncs/0_door_limit30.ncs b/_module/ncs/0_door_limit30.ncs index ee11eb1..ecb3cd2 100644 Binary files a/_module/ncs/0_door_limit30.ncs and b/_module/ncs/0_door_limit30.ncs differ diff --git a/_module/ncs/0_gen_10.ncs b/_module/ncs/0_gen_10.ncs index 5eafe9a..cfacae6 100644 Binary files a/_module/ncs/0_gen_10.ncs and b/_module/ncs/0_gen_10.ncs differ diff --git a/_module/ncs/0_gen_20.ncs b/_module/ncs/0_gen_20.ncs index 7b92e08..51d2ce4 100644 Binary files a/_module/ncs/0_gen_20.ncs and b/_module/ncs/0_gen_20.ncs differ diff --git a/_module/ncs/0_gen_30.ncs b/_module/ncs/0_gen_30.ncs index 4c3db58..1bb3863 100644 Binary files a/_module/ncs/0_gen_30.ncs and b/_module/ncs/0_gen_30.ncs differ diff --git a/_module/ncs/0_gen_40.ncs b/_module/ncs/0_gen_40.ncs index 6f72c99..0c42317 100644 Binary files a/_module/ncs/0_gen_40.ncs and b/_module/ncs/0_gen_40.ncs differ diff --git a/_module/ncs/0_gen_5.ncs b/_module/ncs/0_gen_5.ncs index 23639c5..d23b71e 100644 Binary files a/_module/ncs/0_gen_5.ncs and b/_module/ncs/0_gen_5.ncs differ diff --git a/_module/ncs/0_gen_50.ncs b/_module/ncs/0_gen_50.ncs index 29a6e40..04f810e 100644 Binary files a/_module/ncs/0_gen_50.ncs and b/_module/ncs/0_gen_50.ncs differ diff --git a/_module/ncs/0_gen_60.ncs b/_module/ncs/0_gen_60.ncs index dc386c1..14343b1 100644 Binary files a/_module/ncs/0_gen_60.ncs and b/_module/ncs/0_gen_60.ncs differ diff --git a/_module/ncs/0_gen_70.ncs b/_module/ncs/0_gen_70.ncs index a3eea22..45c1435 100644 Binary files a/_module/ncs/0_gen_70.ncs and b/_module/ncs/0_gen_70.ncs differ diff --git a/_module/ncs/0_gen_loot.ncs b/_module/ncs/0_gen_loot.ncs index 29a6e40..04f810e 100644 Binary files a/_module/ncs/0_gen_loot.ncs and b/_module/ncs/0_gen_loot.ncs differ diff --git a/_module/ncs/0_party_port.ncs b/_module/ncs/0_party_port.ncs index f121f9a..0cb2e0a 100644 Binary files a/_module/ncs/0_party_port.ncs and b/_module/ncs/0_party_port.ncs differ diff --git a/_module/ncs/0_trinity_home.ncs b/_module/ncs/0_trinity_home.ncs index ea5a475..0b3e9c0 100644 Binary files a/_module/ncs/0_trinity_home.ncs and b/_module/ncs/0_trinity_home.ncs differ diff --git a/_module/ncs/1_clean_store.ncs b/_module/ncs/1_clean_store.ncs index ea12faf..69f2f28 100644 Binary files a/_module/ncs/1_clean_store.ncs and b/_module/ncs/1_clean_store.ncs differ diff --git a/_module/ncs/1_loot_unique.ncs b/_module/ncs/1_loot_unique.ncs index 34b24db..4d674e6 100644 Binary files a/_module/ncs/1_loot_unique.ncs and b/_module/ncs/1_loot_unique.ncs differ diff --git a/_module/ncs/2_despawn_loot.ncs b/_module/ncs/2_despawn_loot.ncs index e994c13..9038ae9 100644 Binary files a/_module/ncs/2_despawn_loot.ncs and b/_module/ncs/2_despawn_loot.ncs differ diff --git a/_module/ncs/2_die_portals_ho.ncs b/_module/ncs/2_die_portals_ho.ncs index 3c1280e..370f7b6 100644 Binary files a/_module/ncs/2_die_portals_ho.ncs and b/_module/ncs/2_die_portals_ho.ncs differ diff --git a/_module/ncs/2_explode_death.ncs b/_module/ncs/2_explode_death.ncs index e74c91e..9b756dc 100644 Binary files a/_module/ncs/2_explode_death.ncs and b/_module/ncs/2_explode_death.ncs differ diff --git a/_module/ncs/2_gong_port.ncs b/_module/ncs/2_gong_port.ncs index 481f04e..26b73e7 100644 Binary files a/_module/ncs/2_gong_port.ncs and b/_module/ncs/2_gong_port.ncs differ diff --git a/_module/ncs/2_polar_party1.ncs b/_module/ncs/2_polar_party1.ncs index 27a28b5..fccd503 100644 Binary files a/_module/ncs/2_polar_party1.ncs and b/_module/ncs/2_polar_party1.ncs differ diff --git a/_module/ncs/2_polar_party2.ncs b/_module/ncs/2_polar_party2.ncs index 7d67a2f..9b6994e 100644 Binary files a/_module/ncs/2_polar_party2.ncs and b/_module/ncs/2_polar_party2.ncs differ diff --git a/_module/ncs/2_ship_2k.ncs b/_module/ncs/2_ship_2k.ncs index 5e13915..6a9c1cb 100644 Binary files a/_module/ncs/2_ship_2k.ncs and b/_module/ncs/2_ship_2k.ncs differ diff --git a/_module/ncs/2_ship_gold.ncs b/_module/ncs/2_ship_gold.ncs index 72ffe50..ad3ffc6 100644 Binary files a/_module/ncs/2_ship_gold.ncs and b/_module/ncs/2_ship_gold.ncs differ diff --git a/_module/ncs/2_sit_down.ncs b/_module/ncs/2_sit_down.ncs index 9d356d9..4a7025c 100644 Binary files a/_module/ncs/2_sit_down.ncs and b/_module/ncs/2_sit_down.ncs differ diff --git a/_module/ncs/2_trin_home_01.ncs b/_module/ncs/2_trin_home_01.ncs index bc724c5..d515c80 100644 Binary files a/_module/ncs/2_trin_home_01.ncs and b/_module/ncs/2_trin_home_01.ncs differ diff --git a/_module/ncs/2_trinarena.ncs b/_module/ncs/2_trinarena.ncs index b391903..5c8620c 100644 Binary files a/_module/ncs/2_trinarena.ncs and b/_module/ncs/2_trinarena.ncs differ diff --git a/_module/ncs/9_goblinl_loot.ncs b/_module/ncs/9_goblinl_loot.ncs index 24099d6..83b03ff 100644 Binary files a/_module/ncs/9_goblinl_loot.ncs and b/_module/ncs/9_goblinl_loot.ncs differ diff --git a/_module/ncs/_key.ncs b/_module/ncs/_key.ncs index 090d234..f0f804a 100644 Binary files a/_module/ncs/_key.ncs and b/_module/ncs/_key.ncs differ diff --git a/_module/ncs/aa_removequest.ncs b/_module/ncs/aa_removequest.ncs index bbd37dd..e0ab48d 100644 Binary files a/_module/ncs/aa_removequest.ncs and b/_module/ncs/aa_removequest.ncs differ diff --git a/_module/ncs/asg_alcdesk_01t.ncs b/_module/ncs/asg_alcdesk_01t.ncs index dd53d42..faa2678 100644 Binary files a/_module/ncs/asg_alcdesk_01t.ncs and b/_module/ncs/asg_alcdesk_01t.ncs differ diff --git a/_module/ncs/asg_alcdsk_mat01.ncs b/_module/ncs/asg_alcdsk_mat01.ncs index c77da9c..9729015 100644 Binary files a/_module/ncs/asg_alcdsk_mat01.ncs and b/_module/ncs/asg_alcdsk_mat01.ncs differ diff --git a/_module/ncs/asg_enchan_01t.ncs b/_module/ncs/asg_enchan_01t.ncs index 0a35c46..3d2cb79 100644 Binary files a/_module/ncs/asg_enchan_01t.ncs and b/_module/ncs/asg_enchan_01t.ncs differ diff --git a/_module/ncs/asg_resdesk_06t.ncs b/_module/ncs/asg_resdesk_06t.ncs index 3740e6e..5d00a3f 100644 Binary files a/_module/ncs/asg_resdesk_06t.ncs and b/_module/ncs/asg_resdesk_06t.ncs differ diff --git a/_module/ncs/asg_rul_buldmagi.ncs b/_module/ncs/asg_rul_buldmagi.ncs index c9a566b..665d683 100644 Binary files a/_module/ncs/asg_rul_buldmagi.ncs and b/_module/ncs/asg_rul_buldmagi.ncs differ diff --git a/_module/ncs/asg_rul_micscfx.ncs b/_module/ncs/asg_rul_micscfx.ncs index 1f9c874..ba1d9d6 100644 Binary files a/_module/ncs/asg_rul_micscfx.ncs and b/_module/ncs/asg_rul_micscfx.ncs differ diff --git a/_module/ncs/asg_rul_testmcop.ncs b/_module/ncs/asg_rul_testmcop.ncs index d2b26a5..5616978 100644 Binary files a/_module/ncs/asg_rul_testmcop.ncs and b/_module/ncs/asg_rul_testmcop.ncs differ diff --git a/_module/ncs/asg_tes_alanvtim.ncs b/_module/ncs/asg_tes_alanvtim.ncs index 0b749a7..d52944a 100644 Binary files a/_module/ncs/asg_tes_alanvtim.ncs and b/_module/ncs/asg_tes_alanvtim.ncs differ diff --git a/_module/ncs/asg_tes_alteranv.ncs b/_module/ncs/asg_tes_alteranv.ncs index c4848f1..66a2375 100644 Binary files a/_module/ncs/asg_tes_alteranv.ncs and b/_module/ncs/asg_tes_alteranv.ncs differ diff --git a/_module/ncs/asg_tes_anvaltrc.ncs b/_module/ncs/asg_tes_anvaltrc.ncs index c9cbe2d..a2df92f 100644 Binary files a/_module/ncs/asg_tes_anvaltrc.ncs and b/_module/ncs/asg_tes_anvaltrc.ncs differ diff --git a/_module/ncs/at_ambas_die.ncs b/_module/ncs/at_ambas_die.ncs index 10c98de..92e03a4 100644 Binary files a/_module/ncs/at_ambas_die.ncs and b/_module/ncs/at_ambas_die.ncs differ diff --git a/_module/ncs/at_ambasattack.ncs b/_module/ncs/at_ambasattack.ncs index d859cf8..fe0cd29 100644 Binary files a/_module/ncs/at_ambasattack.ncs and b/_module/ncs/at_ambasattack.ncs differ diff --git a/_module/ncs/at_ass_ship_01.ncs b/_module/ncs/at_ass_ship_01.ncs index a9559cf..f26a632 100644 Binary files a/_module/ncs/at_ass_ship_01.ncs and b/_module/ncs/at_ass_ship_01.ncs differ diff --git a/_module/ncs/at_ass_ship_02.ncs b/_module/ncs/at_ass_ship_02.ncs index dc62465..5d41733 100644 Binary files a/_module/ncs/at_ass_ship_02.ncs and b/_module/ncs/at_ass_ship_02.ncs differ diff --git a/_module/ncs/at_assbook_02.ncs b/_module/ncs/at_assbook_02.ncs index 435e9ac..aaecb1a 100644 Binary files a/_module/ncs/at_assbook_02.ncs and b/_module/ncs/at_assbook_02.ncs differ diff --git a/_module/ncs/at_assbook_03.ncs b/_module/ncs/at_assbook_03.ncs index 0c3a1f8..ee423bd 100644 Binary files a/_module/ncs/at_assbook_03.ncs and b/_module/ncs/at_assbook_03.ncs differ diff --git a/_module/ncs/at_assbook_04.ncs b/_module/ncs/at_assbook_04.ncs index 0b1728f..201cf16 100644 Binary files a/_module/ncs/at_assbook_04.ncs and b/_module/ncs/at_assbook_04.ncs differ diff --git a/_module/ncs/at_assbook_05.ncs b/_module/ncs/at_assbook_05.ncs index a8f1035..90c41d0 100644 Binary files a/_module/ncs/at_assbook_05.ncs and b/_module/ncs/at_assbook_05.ncs differ diff --git a/_module/ncs/at_assbook_06.ncs b/_module/ncs/at_assbook_06.ncs index dd846af..0ffe4b4 100644 Binary files a/_module/ncs/at_assbook_06.ncs and b/_module/ncs/at_assbook_06.ncs differ diff --git a/_module/ncs/at_assbook_07.ncs b/_module/ncs/at_assbook_07.ncs index 303cbfd..3f106f4 100644 Binary files a/_module/ncs/at_assbook_07.ncs and b/_module/ncs/at_assbook_07.ncs differ diff --git a/_module/ncs/at_assbook_08.ncs b/_module/ncs/at_assbook_08.ncs index 59a0f32..ccac9a8 100644 Binary files a/_module/ncs/at_assbook_08.ncs and b/_module/ncs/at_assbook_08.ncs differ diff --git a/_module/ncs/at_assbook_09.ncs b/_module/ncs/at_assbook_09.ncs index c637d00..e379823 100644 Binary files a/_module/ncs/at_assbook_09.ncs and b/_module/ncs/at_assbook_09.ncs differ diff --git a/_module/ncs/at_assbook_1.ncs b/_module/ncs/at_assbook_1.ncs index 9fb4208..7f5e806 100644 Binary files a/_module/ncs/at_assbook_1.ncs and b/_module/ncs/at_assbook_1.ncs differ diff --git a/_module/ncs/at_assbook_10.ncs b/_module/ncs/at_assbook_10.ncs index 12833e0..f84db4c 100644 Binary files a/_module/ncs/at_assbook_10.ncs and b/_module/ncs/at_assbook_10.ncs differ diff --git a/_module/ncs/at_assbook_11.ncs b/_module/ncs/at_assbook_11.ncs index 6f5fe91..d5bb1c2 100644 Binary files a/_module/ncs/at_assbook_11.ncs and b/_module/ncs/at_assbook_11.ncs differ diff --git a/_module/ncs/at_assbook_12.ncs b/_module/ncs/at_assbook_12.ncs index d84af2a..e18c55e 100644 Binary files a/_module/ncs/at_assbook_12.ncs and b/_module/ncs/at_assbook_12.ncs differ diff --git a/_module/ncs/at_assbook_13.ncs b/_module/ncs/at_assbook_13.ncs index 1251448..7bc9490 100644 Binary files a/_module/ncs/at_assbook_13.ncs and b/_module/ncs/at_assbook_13.ncs differ diff --git a/_module/ncs/at_assbook_14.ncs b/_module/ncs/at_assbook_14.ncs index 5b85de6..430ada9 100644 Binary files a/_module/ncs/at_assbook_14.ncs and b/_module/ncs/at_assbook_14.ncs differ diff --git a/_module/ncs/at_asscpt_die.ncs b/_module/ncs/at_asscpt_die.ncs index 6bcd441..2011e03 100644 Binary files a/_module/ncs/at_asscpt_die.ncs and b/_module/ncs/at_asscpt_die.ncs differ diff --git a/_module/ncs/at_asy_noble_die.ncs b/_module/ncs/at_asy_noble_die.ncs index ec353ce..1fe7a95 100644 Binary files a/_module/ncs/at_asy_noble_die.ncs and b/_module/ncs/at_asy_noble_die.ncs differ diff --git a/_module/ncs/at_asy_treas_die.ncs b/_module/ncs/at_asy_treas_die.ncs index de598b1..d0da419 100644 Binary files a/_module/ncs/at_asy_treas_die.ncs and b/_module/ncs/at_asy_treas_die.ncs differ diff --git a/_module/ncs/at_asy_treas_por.ncs b/_module/ncs/at_asy_treas_por.ncs index 832ff04..1f337f8 100644 Binary files a/_module/ncs/at_asy_treas_por.ncs and b/_module/ncs/at_asy_treas_por.ncs differ diff --git a/_module/ncs/at_asyfinal.ncs b/_module/ncs/at_asyfinal.ncs index bd5686a..ca5aa62 100644 Binary files a/_module/ncs/at_asyfinal.ncs and b/_module/ncs/at_asyfinal.ncs differ diff --git a/_module/ncs/at_atlantis_home.ncs b/_module/ncs/at_atlantis_home.ncs index e2af9d8..97bdcbc 100644 Binary files a/_module/ncs/at_atlantis_home.ncs and b/_module/ncs/at_atlantis_home.ncs differ diff --git a/_module/ncs/at_attackpc.ncs b/_module/ncs/at_attackpc.ncs index 48d360d..116ea36 100644 Binary files a/_module/ncs/at_attackpc.ncs and b/_module/ncs/at_attackpc.ncs differ diff --git a/_module/ncs/at_balorjail.ncs b/_module/ncs/at_balorjail.ncs index 06fbaa8..0adf683 100644 Binary files a/_module/ncs/at_balorjail.ncs and b/_module/ncs/at_balorjail.ncs differ diff --git a/_module/ncs/at_bear_01.ncs b/_module/ncs/at_bear_01.ncs index 901db30..d535241 100644 Binary files a/_module/ncs/at_bear_01.ncs and b/_module/ncs/at_bear_01.ncs differ diff --git a/_module/ncs/at_bearship_02.ncs b/_module/ncs/at_bearship_02.ncs index 1be81e1..b295412 100644 Binary files a/_module/ncs/at_bearship_02.ncs and b/_module/ncs/at_bearship_02.ncs differ diff --git a/_module/ncs/at_ch_flaw.ncs b/_module/ncs/at_ch_flaw.ncs index 1ba57ee..1ab20c6 100644 Binary files a/_module/ncs/at_ch_flaw.ncs and b/_module/ncs/at_ch_flaw.ncs differ diff --git a/_module/ncs/at_check_100k.ncs b/_module/ncs/at_check_100k.ncs index 7b16c12..f9ca939 100644 Binary files a/_module/ncs/at_check_100k.ncs and b/_module/ncs/at_check_100k.ncs differ diff --git a/_module/ncs/at_check_meat.ncs b/_module/ncs/at_check_meat.ncs index 7eba055..24a8265 100644 Binary files a/_module/ncs/at_check_meat.ncs and b/_module/ncs/at_check_meat.ncs differ diff --git a/_module/ncs/at_check_sewer.ncs b/_module/ncs/at_check_sewer.ncs index fb5ad42..67f8968 100644 Binary files a/_module/ncs/at_check_sewer.ncs and b/_module/ncs/at_check_sewer.ncs differ diff --git a/_module/ncs/at_checkpelt.ncs b/_module/ncs/at_checkpelt.ncs index 7d1571e..cb4f60c 100644 Binary files a/_module/ncs/at_checkpelt.ncs and b/_module/ncs/at_checkpelt.ncs differ diff --git a/_module/ncs/at_chk_25k.ncs b/_module/ncs/at_chk_25k.ncs index ba4e9dd..52a53f7 100644 Binary files a/_module/ncs/at_chk_25k.ncs and b/_module/ncs/at_chk_25k.ncs differ diff --git a/_module/ncs/at_chk_50k.ncs b/_module/ncs/at_chk_50k.ncs index 64da7ce..fbefe45 100644 Binary files a/_module/ncs/at_chk_50k.ncs and b/_module/ncs/at_chk_50k.ncs differ diff --git a/_module/ncs/at_chk_5k.ncs b/_module/ncs/at_chk_5k.ncs index b16551a..788aa5e 100644 Binary files a/_module/ncs/at_chk_5k.ncs and b/_module/ncs/at_chk_5k.ncs differ diff --git a/_module/ncs/at_chk_broom.ncs b/_module/ncs/at_chk_broom.ncs index 7d81eb1..574d997 100644 Binary files a/_module/ncs/at_chk_broom.ncs and b/_module/ncs/at_chk_broom.ncs differ diff --git a/_module/ncs/at_chk_dragem.ncs b/_module/ncs/at_chk_dragem.ncs index 31331ca..d53e889 100644 Binary files a/_module/ncs/at_chk_dragem.ncs and b/_module/ncs/at_chk_dragem.ncs differ diff --git a/_module/ncs/at_chk_elvis.ncs b/_module/ncs/at_chk_elvis.ncs index 32a6000..ac94dd3 100644 Binary files a/_module/ncs/at_chk_elvis.ncs and b/_module/ncs/at_chk_elvis.ncs differ diff --git a/_module/ncs/at_chk_engtools.ncs b/_module/ncs/at_chk_engtools.ncs index fb47bad..8b56075 100644 Binary files a/_module/ncs/at_chk_engtools.ncs and b/_module/ncs/at_chk_engtools.ncs differ diff --git a/_module/ncs/at_chk_ironore.ncs b/_module/ncs/at_chk_ironore.ncs index fbf77c2..1135daf 100644 Binary files a/_module/ncs/at_chk_ironore.ncs and b/_module/ncs/at_chk_ironore.ncs differ diff --git a/_module/ncs/at_chk_jerkquest.ncs b/_module/ncs/at_chk_jerkquest.ncs index 1f41e88..1428ea7 100644 Binary files a/_module/ncs/at_chk_jerkquest.ncs and b/_module/ncs/at_chk_jerkquest.ncs differ diff --git a/_module/ncs/at_chk_lvl2.ncs b/_module/ncs/at_chk_lvl2.ncs index 4468d7b..f91ebcd 100644 Binary files a/_module/ncs/at_chk_lvl2.ncs and b/_module/ncs/at_chk_lvl2.ncs differ diff --git a/_module/ncs/at_chk_massarmor.ncs b/_module/ncs/at_chk_massarmor.ncs index 8f2929b..2b248ef 100644 Binary files a/_module/ncs/at_chk_massarmor.ncs and b/_module/ncs/at_chk_massarmor.ncs differ diff --git a/_module/ncs/at_chk_masshelm.ncs b/_module/ncs/at_chk_masshelm.ncs index 1ba5a21..f821561 100644 Binary files a/_module/ncs/at_chk_masshelm.ncs and b/_module/ncs/at_chk_masshelm.ncs differ diff --git a/_module/ncs/at_chk_necklace.ncs b/_module/ncs/at_chk_necklace.ncs index a8e8c9d..74a4070 100644 Binary files a/_module/ncs/at_chk_necklace.ncs and b/_module/ncs/at_chk_necklace.ncs differ diff --git a/_module/ncs/at_chk_oldneck.ncs b/_module/ncs/at_chk_oldneck.ncs index 89c5f1e..ff9588b 100644 Binary files a/_module/ncs/at_chk_oldneck.ncs and b/_module/ncs/at_chk_oldneck.ncs differ diff --git a/_module/ncs/at_chk_painting.ncs b/_module/ncs/at_chk_painting.ncs index c72a1b8..cc68b11 100644 Binary files a/_module/ncs/at_chk_painting.ncs and b/_module/ncs/at_chk_painting.ncs differ diff --git a/_module/ncs/at_chk_pickaxe.ncs b/_module/ncs/at_chk_pickaxe.ncs index 7f3150c..0f1b42b 100644 Binary files a/_module/ncs/at_chk_pickaxe.ncs and b/_module/ncs/at_chk_pickaxe.ncs differ diff --git a/_module/ncs/at_chk_puzz.ncs b/_module/ncs/at_chk_puzz.ncs index 4f033c4..7ac4adb 100644 Binary files a/_module/ncs/at_chk_puzz.ncs and b/_module/ncs/at_chk_puzz.ncs differ diff --git a/_module/ncs/at_chk_shieldleg.ncs b/_module/ncs/at_chk_shieldleg.ncs index 2cf01e3..cd0d4a5 100644 Binary files a/_module/ncs/at_chk_shieldleg.ncs and b/_module/ncs/at_chk_shieldleg.ncs differ diff --git a/_module/ncs/at_chk_shovel.ncs b/_module/ncs/at_chk_shovel.ncs index b58eccc..3b6d224 100644 Binary files a/_module/ncs/at_chk_shovel.ncs and b/_module/ncs/at_chk_shovel.ncs differ diff --git a/_module/ncs/at_chk_shsword.ncs b/_module/ncs/at_chk_shsword.ncs index 1e2afec..fd8ce53 100644 Binary files a/_module/ncs/at_chk_shsword.ncs and b/_module/ncs/at_chk_shsword.ncs differ diff --git a/_module/ncs/at_chk_weavfeath.ncs b/_module/ncs/at_chk_weavfeath.ncs index 220069e..abdd5cc 100644 Binary files a/_module/ncs/at_chk_weavfeath.ncs and b/_module/ncs/at_chk_weavfeath.ncs differ diff --git a/_module/ncs/at_chk_wp.ncs b/_module/ncs/at_chk_wp.ncs index ed67b4a..f383d01 100644 Binary files a/_module/ncs/at_chk_wp.ncs and b/_module/ncs/at_chk_wp.ncs differ diff --git a/_module/ncs/at_chk_wphelm.ncs b/_module/ncs/at_chk_wphelm.ncs index ea5e873..e963889 100644 Binary files a/_module/ncs/at_chk_wphelm.ncs and b/_module/ncs/at_chk_wphelm.ncs differ diff --git a/_module/ncs/at_chk_yoga.ncs b/_module/ncs/at_chk_yoga.ncs index 84a5047..f6c9869 100644 Binary files a/_module/ncs/at_chk_yoga.ncs and b/_module/ncs/at_chk_yoga.ncs differ diff --git a/_module/ncs/at_chkjimq1.ncs b/_module/ncs/at_chkjimq1.ncs index 4f033c4..7ac4adb 100644 Binary files a/_module/ncs/at_chkjimq1.ncs and b/_module/ncs/at_chkjimq1.ncs differ diff --git a/_module/ncs/at_chkunenq2.ncs b/_module/ncs/at_chkunenq2.ncs index 3e89d01..0517eff 100644 Binary files a/_module/ncs/at_chkunenq2.ncs and b/_module/ncs/at_chkunenq2.ncs differ diff --git a/_module/ncs/at_ck_arcanemas.ncs b/_module/ncs/at_ck_arcanemas.ncs index 67c17be..76f523b 100644 Binary files a/_module/ncs/at_ck_arcanemas.ncs and b/_module/ncs/at_ck_arcanemas.ncs differ diff --git a/_module/ncs/at_ck_ass_armor.ncs b/_module/ncs/at_ck_ass_armor.ncs index bf708a7..b2b224b 100644 Binary files a/_module/ncs/at_ck_ass_armor.ncs and b/_module/ncs/at_ck_ass_armor.ncs differ diff --git a/_module/ncs/at_ck_robelegend.ncs b/_module/ncs/at_ck_robelegend.ncs index aee3b61..d1084b3 100644 Binary files a/_module/ncs/at_ck_robelegend.ncs and b/_module/ncs/at_ck_robelegend.ncs differ diff --git a/_module/ncs/at_cke_reapboth.ncs b/_module/ncs/at_cke_reapboth.ncs index 9c61239..a48239d 100644 Binary files a/_module/ncs/at_cke_reapboth.ncs and b/_module/ncs/at_cke_reapboth.ncs differ diff --git a/_module/ncs/at_ckeglovesle.ncs b/_module/ncs/at_ckeglovesle.ncs index 3b7e98a..0a250c1 100644 Binary files a/_module/ncs/at_ckeglovesle.ncs and b/_module/ncs/at_ckeglovesle.ncs differ diff --git a/_module/ncs/at_ckemonkhelm.ncs b/_module/ncs/at_ckemonkhelm.ncs index fd1ebba..b6271b6 100644 Binary files a/_module/ncs/at_ckemonkhelm.ncs and b/_module/ncs/at_ckemonkhelm.ncs differ diff --git a/_module/ncs/at_cloaksecret.ncs b/_module/ncs/at_cloaksecret.ncs index fbdce74..112321e 100644 Binary files a/_module/ncs/at_cloaksecret.ncs and b/_module/ncs/at_cloaksecret.ncs differ diff --git a/_module/ncs/at_dickshield.ncs b/_module/ncs/at_dickshield.ncs index c27721d..92f53db 100644 Binary files a/_module/ncs/at_dickshield.ncs and b/_module/ncs/at_dickshield.ncs differ diff --git a/_module/ncs/at_dm_4caves.ncs b/_module/ncs/at_dm_4caves.ncs index 63bb85f..daab3a8 100644 Binary files a/_module/ncs/at_dm_4caves.ncs and b/_module/ncs/at_dm_4caves.ncs differ diff --git a/_module/ncs/at_dm_dragon.ncs b/_module/ncs/at_dm_dragon.ncs index 40d33d0..2947f7d 100644 Binary files a/_module/ncs/at_dm_dragon.ncs and b/_module/ncs/at_dm_dragon.ncs differ diff --git a/_module/ncs/at_dm_dwarf.ncs b/_module/ncs/at_dm_dwarf.ncs index 4573ece..7934f77 100644 Binary files a/_module/ncs/at_dm_dwarf.ncs and b/_module/ncs/at_dm_dwarf.ncs differ diff --git a/_module/ncs/at_dm_fareast.ncs b/_module/ncs/at_dm_fareast.ncs index 21a8031..e38387f 100644 Binary files a/_module/ncs/at_dm_fareast.ncs and b/_module/ncs/at_dm_fareast.ncs differ diff --git a/_module/ncs/at_dm_shogun.ncs b/_module/ncs/at_dm_shogun.ncs index 646a9cc..1a7b232 100644 Binary files a/_module/ncs/at_dm_shogun.ncs and b/_module/ncs/at_dm_shogun.ncs differ diff --git a/_module/ncs/at_dmbadge.ncs b/_module/ncs/at_dmbadge.ncs index a653604..50e0cea 100644 Binary files a/_module/ncs/at_dmbadge.ncs and b/_module/ncs/at_dmbadge.ncs differ diff --git a/_module/ncs/at_dmport32.ncs b/_module/ncs/at_dmport32.ncs index eaaa0a5..fea8f53 100644 Binary files a/_module/ncs/at_dmport32.ncs and b/_module/ncs/at_dmport32.ncs differ diff --git a/_module/ncs/at_doc_01.ncs b/_module/ncs/at_doc_01.ncs index dc0cbf1..ee4258a 100644 Binary files a/_module/ncs/at_doc_01.ncs and b/_module/ncs/at_doc_01.ncs differ diff --git a/_module/ncs/at_doc_02.ncs b/_module/ncs/at_doc_02.ncs index 4b94665..c277df9 100644 Binary files a/_module/ncs/at_doc_02.ncs and b/_module/ncs/at_doc_02.ncs differ diff --git a/_module/ncs/at_docport02.ncs b/_module/ncs/at_docport02.ncs index 5109df0..ee7bdef 100644 Binary files a/_module/ncs/at_docport02.ncs and b/_module/ncs/at_docport02.ncs differ diff --git a/_module/ncs/at_docport03.ncs b/_module/ncs/at_docport03.ncs index b84b3e2..be345e3 100644 Binary files a/_module/ncs/at_docport03.ncs and b/_module/ncs/at_docport03.ncs differ diff --git a/_module/ncs/at_docport4.ncs b/_module/ncs/at_docport4.ncs index 2cf83d5..ff243ab 100644 Binary files a/_module/ncs/at_docport4.ncs and b/_module/ncs/at_docport4.ncs differ diff --git a/_module/ncs/at_dragport_01.ncs b/_module/ncs/at_dragport_01.ncs index 9880cef..53513e9 100644 Binary files a/_module/ncs/at_dragport_01.ncs and b/_module/ncs/at_dragport_01.ncs differ diff --git a/_module/ncs/at_dragport_02.ncs b/_module/ncs/at_dragport_02.ncs index d78a4ad..59f5f36 100644 Binary files a/_module/ncs/at_dragport_02.ncs and b/_module/ncs/at_dragport_02.ncs differ diff --git a/_module/ncs/at_dreamport.ncs b/_module/ncs/at_dreamport.ncs index a6ea966..908fb35 100644 Binary files a/_module/ncs/at_dreamport.ncs and b/_module/ncs/at_dreamport.ncs differ diff --git a/_module/ncs/at_ent_dbi_room.ncs b/_module/ncs/at_ent_dbi_room.ncs new file mode 100644 index 0000000..b68899e Binary files /dev/null and b/_module/ncs/at_ent_dbi_room.ncs differ diff --git a/_module/ncs/at_enter_fists.ncs b/_module/ncs/at_enter_fists.ncs index 70e6a9f..6a60594 100644 Binary files a/_module/ncs/at_enter_fists.ncs and b/_module/ncs/at_enter_fists.ncs differ diff --git a/_module/ncs/at_enter_issac.ncs b/_module/ncs/at_enter_issac.ncs index 1f470a9..ed4b392 100644 Binary files a/_module/ncs/at_enter_issac.ncs and b/_module/ncs/at_enter_issac.ncs differ diff --git a/_module/ncs/at_enter_pc_room.ncs b/_module/ncs/at_enter_pc_room.ncs new file mode 100644 index 0000000..a9483ab Binary files /dev/null and b/_module/ncs/at_enter_pc_room.ncs differ diff --git a/_module/ncs/at_farm_joe_01.ncs b/_module/ncs/at_farm_joe_01.ncs index 9adca0b..11a7d28 100644 Binary files a/_module/ncs/at_farm_joe_01.ncs and b/_module/ncs/at_farm_joe_01.ncs differ diff --git a/_module/ncs/at_fists_key.ncs b/_module/ncs/at_fists_key.ncs index 153f44b..13279f2 100644 Binary files a/_module/ncs/at_fists_key.ncs and b/_module/ncs/at_fists_key.ncs differ diff --git a/_module/ncs/at_forge_staff.ncs b/_module/ncs/at_forge_staff.ncs index aae76e9..2d35958 100644 Binary files a/_module/ncs/at_forge_staff.ncs and b/_module/ncs/at_forge_staff.ncs differ diff --git a/_module/ncs/at_give1mill.ncs b/_module/ncs/at_give1mill.ncs index b1c0d7b..9d48dd3 100644 Binary files a/_module/ncs/at_give1mill.ncs and b/_module/ncs/at_give1mill.ncs differ diff --git a/_module/ncs/at_give2k.ncs b/_module/ncs/at_give2k.ncs index 2150a0a..f15913d 100644 Binary files a/_module/ncs/at_give2k.ncs and b/_module/ncs/at_give2k.ncs differ diff --git a/_module/ncs/at_give_300k.ncs b/_module/ncs/at_give_300k.ncs index 801c0ec..035333e 100644 Binary files a/_module/ncs/at_give_300k.ncs and b/_module/ncs/at_give_300k.ncs differ diff --git a/_module/ncs/at_give_50kxp.ncs b/_module/ncs/at_give_50kxp.ncs index 948e6e8..31f8ddf 100644 Binary files a/_module/ncs/at_give_50kxp.ncs and b/_module/ncs/at_give_50kxp.ncs differ diff --git a/_module/ncs/at_give_retrunst.ncs b/_module/ncs/at_give_retrunst.ncs index 63c0b8e..8dd5e04 100644 Binary files a/_module/ncs/at_give_retrunst.ncs and b/_module/ncs/at_give_retrunst.ncs differ diff --git a/_module/ncs/at_giveelvis.ncs b/_module/ncs/at_giveelvis.ncs index 06f26b4..7255f02 100644 Binary files a/_module/ncs/at_giveelvis.ncs and b/_module/ncs/at_giveelvis.ncs differ diff --git a/_module/ncs/at_gnome_death.ncs b/_module/ncs/at_gnome_death.ncs index 777e8f5..02857cf 100644 Binary files a/_module/ncs/at_gnome_death.ncs and b/_module/ncs/at_gnome_death.ncs differ diff --git a/_module/ncs/at_gv_dragem.ncs b/_module/ncs/at_gv_dragem.ncs index 9a8fdb3..a7b177c 100644 Binary files a/_module/ncs/at_gv_dragem.ncs and b/_module/ncs/at_gv_dragem.ncs differ diff --git a/_module/ncs/at_hall1.ncs b/_module/ncs/at_hall1.ncs index 39edfe2..08e6c15 100644 Binary files a/_module/ncs/at_hall1.ncs and b/_module/ncs/at_hall1.ncs differ diff --git a/_module/ncs/at_hall2.ncs b/_module/ncs/at_hall2.ncs index f81ebf1..55215f8 100644 Binary files a/_module/ncs/at_hall2.ncs and b/_module/ncs/at_hall2.ncs differ diff --git a/_module/ncs/at_hall3.ncs b/_module/ncs/at_hall3.ncs index 60f8e86..427a9df 100644 Binary files a/_module/ncs/at_hall3.ncs and b/_module/ncs/at_hall3.ncs differ diff --git a/_module/ncs/at_hall4.ncs b/_module/ncs/at_hall4.ncs index 208d58f..1f894c2 100644 Binary files a/_module/ncs/at_hall4.ncs and b/_module/ncs/at_hall4.ncs differ diff --git a/_module/ncs/at_hall5.ncs b/_module/ncs/at_hall5.ncs index 6909b36..68a6a19 100644 Binary files a/_module/ncs/at_hall5.ncs and b/_module/ncs/at_hall5.ncs differ diff --git a/_module/ncs/at_hall6.ncs b/_module/ncs/at_hall6.ncs index cb0579f..8186514 100644 Binary files a/_module/ncs/at_hall6.ncs and b/_module/ncs/at_hall6.ncs differ diff --git a/_module/ncs/at_hall7.ncs b/_module/ncs/at_hall7.ncs index 6526546..9176ec2 100644 Binary files a/_module/ncs/at_hall7.ncs and b/_module/ncs/at_hall7.ncs differ diff --git a/_module/ncs/at_hall8.ncs b/_module/ncs/at_hall8.ncs index bf9b657..ec07adc 100644 Binary files a/_module/ncs/at_hall8.ncs and b/_module/ncs/at_hall8.ncs differ diff --git a/_module/ncs/at_hall9.ncs b/_module/ncs/at_hall9.ncs index 7cc78d8..37ce063 100644 Binary files a/_module/ncs/at_hall9.ncs and b/_module/ncs/at_hall9.ncs differ diff --git a/_module/ncs/at_inferno_01.ncs b/_module/ncs/at_inferno_01.ncs index e9826a6..9c3a2d7 100644 Binary files a/_module/ncs/at_inferno_01.ncs and b/_module/ncs/at_inferno_01.ncs differ diff --git a/_module/ncs/at_inferno_02.ncs b/_module/ncs/at_inferno_02.ncs index 1bf303a..21149dd 100644 Binary files a/_module/ncs/at_inferno_02.ncs and b/_module/ncs/at_inferno_02.ncs differ diff --git a/_module/ncs/at_inferno_03.ncs b/_module/ncs/at_inferno_03.ncs index 3015cc8..1885e9a 100644 Binary files a/_module/ncs/at_inferno_03.ncs and b/_module/ncs/at_inferno_03.ncs differ diff --git a/_module/ncs/at_inferno_04.ncs b/_module/ncs/at_inferno_04.ncs index e010f71..275e1cf 100644 Binary files a/_module/ncs/at_inferno_04.ncs and b/_module/ncs/at_inferno_04.ncs differ diff --git a/_module/ncs/at_inferno_enter.ncs b/_module/ncs/at_inferno_enter.ncs index 24794cb..8cf7276 100644 Binary files a/_module/ncs/at_inferno_enter.ncs and b/_module/ncs/at_inferno_enter.ncs differ diff --git a/_module/ncs/at_iron_01.ncs b/_module/ncs/at_iron_01.ncs index 93fa9c6..52f5605 100644 Binary files a/_module/ncs/at_iron_01.ncs and b/_module/ncs/at_iron_01.ncs differ diff --git a/_module/ncs/at_iron_02.ncs b/_module/ncs/at_iron_02.ncs index 9aaf5d6..ec54ce0 100644 Binary files a/_module/ncs/at_iron_02.ncs and b/_module/ncs/at_iron_02.ncs differ diff --git a/_module/ncs/at_iron_03.ncs b/_module/ncs/at_iron_03.ncs index 9824bb6..db63b50 100644 Binary files a/_module/ncs/at_iron_03.ncs and b/_module/ncs/at_iron_03.ncs differ diff --git a/_module/ncs/at_iron_04.ncs b/_module/ncs/at_iron_04.ncs index 1134eb1..6c6e8dd 100644 Binary files a/_module/ncs/at_iron_04.ncs and b/_module/ncs/at_iron_04.ncs differ diff --git a/_module/ncs/at_iron_05.ncs b/_module/ncs/at_iron_05.ncs index 542c6c3..13308a1 100644 Binary files a/_module/ncs/at_iron_05.ncs and b/_module/ncs/at_iron_05.ncs differ diff --git a/_module/ncs/at_iron_06.ncs b/_module/ncs/at_iron_06.ncs index 20297c2..41613ea 100644 Binary files a/_module/ncs/at_iron_06.ncs and b/_module/ncs/at_iron_06.ncs differ diff --git a/_module/ncs/at_iron_07.ncs b/_module/ncs/at_iron_07.ncs index 971197a..a525f79 100644 Binary files a/_module/ncs/at_iron_07.ncs and b/_module/ncs/at_iron_07.ncs differ diff --git a/_module/ncs/at_iron_08.ncs b/_module/ncs/at_iron_08.ncs index 460f79a..a865093 100644 Binary files a/_module/ncs/at_iron_08.ncs and b/_module/ncs/at_iron_08.ncs differ diff --git a/_module/ncs/at_iron_port_09.ncs b/_module/ncs/at_iron_port_09.ncs index 923798d..5865ebc 100644 Binary files a/_module/ncs/at_iron_port_09.ncs and b/_module/ncs/at_iron_port_09.ncs differ diff --git a/_module/ncs/at_issac_01.ncs b/_module/ncs/at_issac_01.ncs index 7b72108..ad847ab 100644 Binary files a/_module/ncs/at_issac_01.ncs and b/_module/ncs/at_issac_01.ncs differ diff --git a/_module/ncs/at_issac_02.ncs b/_module/ncs/at_issac_02.ncs index 6f9f96c..2a7c7f9 100644 Binary files a/_module/ncs/at_issac_02.ncs and b/_module/ncs/at_issac_02.ncs differ diff --git a/_module/ncs/at_issac_03.ncs b/_module/ncs/at_issac_03.ncs index 4ab9115..585c7f4 100644 Binary files a/_module/ncs/at_issac_03.ncs and b/_module/ncs/at_issac_03.ncs differ diff --git a/_module/ncs/at_issac_04.ncs b/_module/ncs/at_issac_04.ncs index cda7d37..ec72d31 100644 Binary files a/_module/ncs/at_issac_04.ncs and b/_module/ncs/at_issac_04.ncs differ diff --git a/_module/ncs/at_jimlast.ncs b/_module/ncs/at_jimlast.ncs index a8b3f9f..4424963 100644 Binary files a/_module/ncs/at_jimlast.ncs and b/_module/ncs/at_jimlast.ncs differ diff --git a/_module/ncs/at_jimmybook.ncs b/_module/ncs/at_jimmybook.ncs index e209dbc..1bbfba1 100644 Binary files a/_module/ncs/at_jimmybook.ncs and b/_module/ncs/at_jimmybook.ncs differ diff --git a/_module/ncs/at_keg.ncs b/_module/ncs/at_keg.ncs index 038f3e1..3e47d4d 100644 Binary files a/_module/ncs/at_keg.ncs and b/_module/ncs/at_keg.ncs differ diff --git a/_module/ncs/at_legionport7.ncs b/_module/ncs/at_legionport7.ncs index 311d0ef..18e3153 100644 Binary files a/_module/ncs/at_legionport7.ncs and b/_module/ncs/at_legionport7.ncs differ diff --git a/_module/ncs/at_lil_fight_die.ncs b/_module/ncs/at_lil_fight_die.ncs index 2e475ad..5e90243 100644 Binary files a/_module/ncs/at_lil_fight_die.ncs and b/_module/ncs/at_lil_fight_die.ncs differ diff --git a/_module/ncs/at_lucephdie.ncs b/_module/ncs/at_lucephdie.ncs index bf1f1bc..93c926b 100644 Binary files a/_module/ncs/at_lucephdie.ncs and b/_module/ncs/at_lucephdie.ncs differ diff --git a/_module/ncs/at_make_mnystone.ncs b/_module/ncs/at_make_mnystone.ncs index 138291a..0ad8bd5 100644 Binary files a/_module/ncs/at_make_mnystone.ncs and b/_module/ncs/at_make_mnystone.ncs differ diff --git a/_module/ncs/at_make_ozkey.ncs b/_module/ncs/at_make_ozkey.ncs index b8ecb5b..6509947 100644 Binary files a/_module/ncs/at_make_ozkey.ncs and b/_module/ncs/at_make_ozkey.ncs differ diff --git a/_module/ncs/at_mared03.ncs b/_module/ncs/at_mared03.ncs index 16eeeb5..e73f537 100644 Binary files a/_module/ncs/at_mared03.ncs and b/_module/ncs/at_mared03.ncs differ diff --git a/_module/ncs/at_mick_100.ncs b/_module/ncs/at_mick_100.ncs index 07cffc7..1ce8a5c 100644 Binary files a/_module/ncs/at_mick_100.ncs and b/_module/ncs/at_mick_100.ncs differ diff --git a/_module/ncs/at_mosquito_spaw.ncs b/_module/ncs/at_mosquito_spaw.ncs index e6959dc..68d3c43 100644 Binary files a/_module/ncs/at_mosquito_spaw.ncs and b/_module/ncs/at_mosquito_spaw.ncs differ diff --git a/_module/ncs/at_move_asscpt.ncs b/_module/ncs/at_move_asscpt.ncs index 90a827a..b966448 100644 Binary files a/_module/ncs/at_move_asscpt.ncs and b/_module/ncs/at_move_asscpt.ncs differ diff --git a/_module/ncs/at_mystic_party.ncs b/_module/ncs/at_mystic_party.ncs index fdb9aaf..d496ca9 100644 Binary files a/_module/ncs/at_mystic_party.ncs and b/_module/ncs/at_mystic_party.ncs differ diff --git a/_module/ncs/at_nikiperceive.ncs b/_module/ncs/at_nikiperceive.ncs index 55deaef..bed8b92 100644 Binary files a/_module/ncs/at_nikiperceive.ncs and b/_module/ncs/at_nikiperceive.ncs differ diff --git a/_module/ncs/at_northernport.ncs b/_module/ncs/at_northernport.ncs index f6b88c8..40ada96 100644 Binary files a/_module/ncs/at_northernport.ncs and b/_module/ncs/at_northernport.ncs differ diff --git a/_module/ncs/at_nude_die.ncs b/_module/ncs/at_nude_die.ncs index 890c169..203423f 100644 Binary files a/_module/ncs/at_nude_die.ncs and b/_module/ncs/at_nude_die.ncs differ diff --git a/_module/ncs/at_nude_perc.ncs b/_module/ncs/at_nude_perc.ncs index a2ec127..62c5b68 100644 Binary files a/_module/ncs/at_nude_perc.ncs and b/_module/ncs/at_nude_perc.ncs differ diff --git a/_module/ncs/at_oz_lowerbook.ncs b/_module/ncs/at_oz_lowerbook.ncs index 8a5bac1..1dea4c5 100644 Binary files a/_module/ncs/at_oz_lowerbook.ncs and b/_module/ncs/at_oz_lowerbook.ncs differ diff --git a/_module/ncs/at_oz_port_02.ncs b/_module/ncs/at_oz_port_02.ncs index a961d57..2381bc3 100644 Binary files a/_module/ncs/at_oz_port_02.ncs and b/_module/ncs/at_oz_port_02.ncs differ diff --git a/_module/ncs/at_oz_port_trin.ncs b/_module/ncs/at_oz_port_trin.ncs index ad976ee..951e236 100644 Binary files a/_module/ncs/at_oz_port_trin.ncs and b/_module/ncs/at_oz_port_trin.ncs differ diff --git a/_module/ncs/at_oz_return.ncs b/_module/ncs/at_oz_return.ncs index 253ee2a..ab4593d 100644 Binary files a/_module/ncs/at_oz_return.ncs and b/_module/ncs/at_oz_return.ncs differ diff --git a/_module/ncs/at_oz_ruinbook.ncs b/_module/ncs/at_oz_ruinbook.ncs index 59b96d5..6901940 100644 Binary files a/_module/ncs/at_oz_ruinbook.ncs and b/_module/ncs/at_oz_ruinbook.ncs differ diff --git a/_module/ncs/at_ozcase.ncs b/_module/ncs/at_ozcase.ncs index 7027efd..4528cf1 100644 Binary files a/_module/ncs/at_ozcase.ncs and b/_module/ncs/at_ozcase.ncs differ diff --git a/_module/ncs/at_ozking_die.ncs b/_module/ncs/at_ozking_die.ncs index 149c222..d101356 100644 Binary files a/_module/ncs/at_ozking_die.ncs and b/_module/ncs/at_ozking_die.ncs differ diff --git a/_module/ncs/at_ozknight_spaw.ncs b/_module/ncs/at_ozknight_spaw.ncs index 680bb46..0f50603 100644 Binary files a/_module/ncs/at_ozknight_spaw.ncs and b/_module/ncs/at_ozknight_spaw.ncs differ diff --git a/_module/ncs/at_port_recruit.ncs b/_module/ncs/at_port_recruit.ncs index 38932da..5eccae6 100644 Binary files a/_module/ncs/at_port_recruit.ncs and b/_module/ncs/at_port_recruit.ncs differ diff --git a/_module/ncs/at_portbreth_01.ncs b/_module/ncs/at_portbreth_01.ncs index 0e140c6..bedb6c9 100644 Binary files a/_module/ncs/at_portbreth_01.ncs and b/_module/ncs/at_portbreth_01.ncs differ diff --git a/_module/ncs/at_quito_die2.ncs b/_module/ncs/at_quito_die2.ncs index 72cf094..6d1197d 100644 Binary files a/_module/ncs/at_quito_die2.ncs and b/_module/ncs/at_quito_die2.ncs differ diff --git a/_module/ncs/at_reb_trialspor.ncs b/_module/ncs/at_reb_trialspor.ncs index 8ed31fc..fee2bc6 100644 Binary files a/_module/ncs/at_reb_trialspor.ncs and b/_module/ncs/at_reb_trialspor.ncs differ diff --git a/_module/ncs/at_recruit_boot.ncs b/_module/ncs/at_recruit_boot.ncs index daf4e60..dbdfaf5 100644 Binary files a/_module/ncs/at_recruit_boot.ncs and b/_module/ncs/at_recruit_boot.ncs differ diff --git a/_module/ncs/at_recruit_chk.ncs b/_module/ncs/at_recruit_chk.ncs index 1c370ab..309ba87 100644 Binary files a/_module/ncs/at_recruit_chk.ncs and b/_module/ncs/at_recruit_chk.ncs differ diff --git a/_module/ncs/at_reddragon.ncs b/_module/ncs/at_reddragon.ncs index 6554bac..f190a7c 100644 Binary files a/_module/ncs/at_reddragon.ncs and b/_module/ncs/at_reddragon.ncs differ diff --git a/_module/ncs/at_retire_attack.ncs b/_module/ncs/at_retire_attack.ncs index 48d360d..116ea36 100644 Binary files a/_module/ncs/at_retire_attack.ncs and b/_module/ncs/at_retire_attack.ncs differ diff --git a/_module/ncs/at_returncoast.ncs b/_module/ncs/at_returncoast.ncs index d91d867..b007126 100644 Binary files a/_module/ncs/at_returncoast.ncs and b/_module/ncs/at_returncoast.ncs differ diff --git a/_module/ncs/at_secret_king.ncs b/_module/ncs/at_secret_king.ncs index 90d5214..7b67fec 100644 Binary files a/_module/ncs/at_secret_king.ncs and b/_module/ncs/at_secret_king.ncs differ diff --git a/_module/ncs/at_secret_king2.ncs b/_module/ncs/at_secret_king2.ncs index ad664cc..2d882a1 100644 Binary files a/_module/ncs/at_secret_king2.ncs and b/_module/ncs/at_secret_king2.ncs differ diff --git a/_module/ncs/at_secret_sexy1.ncs b/_module/ncs/at_secret_sexy1.ncs index 6bd854c..7d52505 100644 Binary files a/_module/ncs/at_secret_sexy1.ncs and b/_module/ncs/at_secret_sexy1.ncs differ diff --git a/_module/ncs/at_secret_sexy2.ncs b/_module/ncs/at_secret_sexy2.ncs index 1d3ba99..f79a80f 100644 Binary files a/_module/ncs/at_secret_sexy2.ncs and b/_module/ncs/at_secret_sexy2.ncs differ diff --git a/_module/ncs/at_secret_tk.ncs b/_module/ncs/at_secret_tk.ncs index 573c4e9..62c0d15 100644 Binary files a/_module/ncs/at_secret_tk.ncs and b/_module/ncs/at_secret_tk.ncs differ diff --git a/_module/ncs/at_secret_tk2.ncs b/_module/ncs/at_secret_tk2.ncs index a5f4913..ffb3dbb 100644 Binary files a/_module/ncs/at_secret_tk2.ncs and b/_module/ncs/at_secret_tk2.ncs differ diff --git a/_module/ncs/at_sen_weap_key.ncs b/_module/ncs/at_sen_weap_key.ncs index 542e9e5..214acfa 100644 Binary files a/_module/ncs/at_sen_weap_key.ncs and b/_module/ncs/at_sen_weap_key.ncs differ diff --git a/_module/ncs/at_sewers_1.ncs b/_module/ncs/at_sewers_1.ncs index 142bef2..f752814 100644 Binary files a/_module/ncs/at_sewers_1.ncs and b/_module/ncs/at_sewers_1.ncs differ diff --git a/_module/ncs/at_sewers_2.ncs b/_module/ncs/at_sewers_2.ncs index b1fb949..5169f0c 100644 Binary files a/_module/ncs/at_sewers_2.ncs and b/_module/ncs/at_sewers_2.ncs differ diff --git a/_module/ncs/at_sl_chk_grail.ncs b/_module/ncs/at_sl_chk_grail.ncs index cac5754..9b6a76e 100644 Binary files a/_module/ncs/at_sl_chk_grail.ncs and b/_module/ncs/at_sl_chk_grail.ncs differ diff --git a/_module/ncs/at_spawn_ironbos.ncs b/_module/ncs/at_spawn_ironbos.ncs index ad982ca..9329810 100644 Binary files a/_module/ncs/at_spawn_ironbos.ncs and b/_module/ncs/at_spawn_ironbos.ncs differ diff --git a/_module/ncs/at_take50kxp.ncs b/_module/ncs/at_take50kxp.ncs index d3eb22c..9cc3a37 100644 Binary files a/_module/ncs/at_take50kxp.ncs and b/_module/ncs/at_take50kxp.ncs differ diff --git a/_module/ncs/at_take_ass_head.ncs b/_module/ncs/at_take_ass_head.ncs index e6a3c71..f36dc2a 100644 Binary files a/_module/ncs/at_take_ass_head.ncs and b/_module/ncs/at_take_ass_head.ncs differ diff --git a/_module/ncs/at_take_broom.ncs b/_module/ncs/at_take_broom.ncs index 7248525..e99a158 100644 Binary files a/_module/ncs/at_take_broom.ncs and b/_module/ncs/at_take_broom.ncs differ diff --git a/_module/ncs/at_take_engtools.ncs b/_module/ncs/at_take_engtools.ncs index e70f137..b216071 100644 Binary files a/_module/ncs/at_take_engtools.ncs and b/_module/ncs/at_take_engtools.ncs differ diff --git a/_module/ncs/at_take_grails.ncs b/_module/ncs/at_take_grails.ncs index db32131..b995059 100644 Binary files a/_module/ncs/at_take_grails.ncs and b/_module/ncs/at_take_grails.ncs differ diff --git a/_module/ncs/at_take_meat.ncs b/_module/ncs/at_take_meat.ncs index 5382d9b..b64e06d 100644 Binary files a/_module/ncs/at_take_meat.ncs and b/_module/ncs/at_take_meat.ncs differ diff --git a/_module/ncs/at_take_painting.ncs b/_module/ncs/at_take_painting.ncs index 1fae190..5704771 100644 Binary files a/_module/ncs/at_take_painting.ncs and b/_module/ncs/at_take_painting.ncs differ diff --git a/_module/ncs/at_take_pelt.ncs b/_module/ncs/at_take_pelt.ncs index f1aa634..3d37100 100644 Binary files a/_module/ncs/at_take_pelt.ncs and b/_module/ncs/at_take_pelt.ncs differ diff --git a/_module/ncs/at_take_pickaxe.ncs b/_module/ncs/at_take_pickaxe.ncs index 3c36a50..454e4d2 100644 Binary files a/_module/ncs/at_take_pickaxe.ncs and b/_module/ncs/at_take_pickaxe.ncs differ diff --git a/_module/ncs/at_take_shieldle.ncs b/_module/ncs/at_take_shieldle.ncs index 2d7bd23..08b10f4 100644 Binary files a/_module/ncs/at_take_shieldle.ncs and b/_module/ncs/at_take_shieldle.ncs differ diff --git a/_module/ncs/at_take_shovel.ncs b/_module/ncs/at_take_shovel.ncs index a8dd148..05a0327 100644 Binary files a/_module/ncs/at_take_shovel.ncs and b/_module/ncs/at_take_shovel.ncs differ diff --git a/_module/ncs/at_take_shsword.ncs b/_module/ncs/at_take_shsword.ncs index 397a3c2..437adcf 100644 Binary files a/_module/ncs/at_take_shsword.ncs and b/_module/ncs/at_take_shsword.ncs differ diff --git a/_module/ncs/at_takexp.ncs b/_module/ncs/at_takexp.ncs index 7fc277e..6a3e2ef 100644 Binary files a/_module/ncs/at_takexp.ncs and b/_module/ncs/at_takexp.ncs differ diff --git a/_module/ncs/at_temple.ncs b/_module/ncs/at_temple.ncs index 7a7c1b0..cf6a40d 100644 Binary files a/_module/ncs/at_temple.ncs and b/_module/ncs/at_temple.ncs differ diff --git a/_module/ncs/at_tk_arcanemas.ncs b/_module/ncs/at_tk_arcanemas.ncs index f125b85..7231735 100644 Binary files a/_module/ncs/at_tk_arcanemas.ncs and b/_module/ncs/at_tk_arcanemas.ncs differ diff --git a/_module/ncs/at_tk_flaw.ncs b/_module/ncs/at_tk_flaw.ncs index db899c1..99244a4 100644 Binary files a/_module/ncs/at_tk_flaw.ncs and b/_module/ncs/at_tk_flaw.ncs differ diff --git a/_module/ncs/at_tk_massarmor.ncs b/_module/ncs/at_tk_massarmor.ncs index 03ed81d..7fe95a9 100644 Binary files a/_module/ncs/at_tk_massarmor.ncs and b/_module/ncs/at_tk_massarmor.ncs differ diff --git a/_module/ncs/at_tk_masshelm.ncs b/_module/ncs/at_tk_masshelm.ncs index 4217f4d..98c3e44 100644 Binary files a/_module/ncs/at_tk_masshelm.ncs and b/_module/ncs/at_tk_masshelm.ncs differ diff --git a/_module/ncs/at_tk_wp.ncs b/_module/ncs/at_tk_wp.ncs index b351153..66ac528 100644 Binary files a/_module/ncs/at_tk_wp.ncs and b/_module/ncs/at_tk_wp.ncs differ diff --git a/_module/ncs/at_tk_wphelm.ncs b/_module/ncs/at_tk_wphelm.ncs index e2a02cf..ab2c379 100644 Binary files a/_module/ncs/at_tk_wphelm.ncs and b/_module/ncs/at_tk_wphelm.ncs differ diff --git a/_module/ncs/at_tke_dreamcatc.ncs b/_module/ncs/at_tke_dreamcatc.ncs index f8dfb26..93d1086 100644 Binary files a/_module/ncs/at_tke_dreamcatc.ncs and b/_module/ncs/at_tke_dreamcatc.ncs differ diff --git a/_module/ncs/at_tke_glovesle.ncs b/_module/ncs/at_tke_glovesle.ncs index 68df3e9..97311e7 100644 Binary files a/_module/ncs/at_tke_glovesle.ncs and b/_module/ncs/at_tke_glovesle.ncs differ diff --git a/_module/ncs/at_tke_monkhelm.ncs b/_module/ncs/at_tke_monkhelm.ncs index 5e45547..92a78c5 100644 Binary files a/_module/ncs/at_tke_monkhelm.ncs and b/_module/ncs/at_tke_monkhelm.ncs differ diff --git a/_module/ncs/at_tke_robelegen.ncs b/_module/ncs/at_tke_robelegen.ncs index e921b9c..54fec93 100644 Binary files a/_module/ncs/at_tke_robelegen.ncs and b/_module/ncs/at_tke_robelegen.ncs differ diff --git a/_module/ncs/at_tke_rpboth.ncs b/_module/ncs/at_tke_rpboth.ncs index f6d7141..e3a4ebc 100644 Binary files a/_module/ncs/at_tke_rpboth.ncs and b/_module/ncs/at_tke_rpboth.ncs differ diff --git a/_module/ncs/at_tke_scrolls.ncs b/_module/ncs/at_tke_scrolls.ncs index 4b10bb5..4d09b7a 100644 Binary files a/_module/ncs/at_tke_scrolls.ncs and b/_module/ncs/at_tke_scrolls.ncs differ diff --git a/_module/ncs/at_tkeass_armor.ncs b/_module/ncs/at_tkeass_armor.ncs index fac8dff..496c9fd 100644 Binary files a/_module/ncs/at_tkeass_armor.ncs and b/_module/ncs/at_tkeass_armor.ncs differ diff --git a/_module/ncs/at_tmlord_dead.ncs b/_module/ncs/at_tmlord_dead.ncs index d4b7d1c..c102b58 100644 Binary files a/_module/ncs/at_tmlord_dead.ncs and b/_module/ncs/at_tmlord_dead.ncs differ diff --git a/_module/ncs/at_to_fareast.ncs b/_module/ncs/at_to_fareast.ncs index 59f62a5..0de4746 100644 Binary files a/_module/ncs/at_to_fareast.ncs and b/_module/ncs/at_to_fareast.ncs differ diff --git a/_module/ncs/at_to_repent2.ncs b/_module/ncs/at_to_repent2.ncs index 7de48c5..4d8cebe 100644 Binary files a/_module/ncs/at_to_repent2.ncs and b/_module/ncs/at_to_repent2.ncs differ diff --git a/_module/ncs/at_treamas_attac.ncs b/_module/ncs/at_treamas_attac.ncs index 48d360d..116ea36 100644 Binary files a/_module/ncs/at_treamas_attac.ncs and b/_module/ncs/at_treamas_attac.ncs differ diff --git a/_module/ncs/at_treas_mas_die.ncs b/_module/ncs/at_treas_mas_die.ncs index ffa9557..d16c795 100644 Binary files a/_module/ncs/at_treas_mas_die.ncs and b/_module/ncs/at_treas_mas_die.ncs differ diff --git a/_module/ncs/at_tree_port01.ncs b/_module/ncs/at_tree_port01.ncs index b117cb4..a6476d5 100644 Binary files a/_module/ncs/at_tree_port01.ncs and b/_module/ncs/at_tree_port01.ncs differ diff --git a/_module/ncs/at_treeport_03.ncs b/_module/ncs/at_treeport_03.ncs index b6067c3..2232b56 100644 Binary files a/_module/ncs/at_treeport_03.ncs and b/_module/ncs/at_treeport_03.ncs differ diff --git a/_module/ncs/at_treeport_04.ncs b/_module/ncs/at_treeport_04.ncs index dcd8193..169f7fa 100644 Binary files a/_module/ncs/at_treeport_04.ncs and b/_module/ncs/at_treeport_04.ncs differ diff --git a/_module/ncs/at_treeport_05.ncs b/_module/ncs/at_treeport_05.ncs index c4efba3..d15b19a 100644 Binary files a/_module/ncs/at_treeport_05.ncs and b/_module/ncs/at_treeport_05.ncs differ diff --git a/_module/ncs/at_treeport_06.ncs b/_module/ncs/at_treeport_06.ncs index 13e00ac..5053f85 100644 Binary files a/_module/ncs/at_treeport_06.ncs and b/_module/ncs/at_treeport_06.ncs differ diff --git a/_module/ncs/at_treeport_07.ncs b/_module/ncs/at_treeport_07.ncs index 3036daf..3b4e909 100644 Binary files a/_module/ncs/at_treeport_07.ncs and b/_module/ncs/at_treeport_07.ncs differ diff --git a/_module/ncs/at_tulls_chk_lvl.ncs b/_module/ncs/at_tulls_chk_lvl.ncs index 05d4802..4966362 100644 Binary files a/_module/ncs/at_tulls_chk_lvl.ncs and b/_module/ncs/at_tulls_chk_lvl.ncs differ diff --git a/_module/ncs/at_yoga3.ncs b/_module/ncs/at_yoga3.ncs index 1aadef1..7f668b0 100644 Binary files a/_module/ncs/at_yoga3.ncs and b/_module/ncs/at_yoga3.ncs differ diff --git a/_module/ncs/atport_to_ma.ncs b/_module/ncs/atport_to_ma.ncs index 1532ca6..860f0cb 100644 Binary files a/_module/ncs/atport_to_ma.ncs and b/_module/ncs/atport_to_ma.ncs differ diff --git a/_module/ncs/aw_anitcheat2.ncs b/_module/ncs/aw_anitcheat2.ncs index d2b4d5f..132d0e5 100644 Binary files a/_module/ncs/aw_anitcheat2.ncs and b/_module/ncs/aw_anitcheat2.ncs differ diff --git a/_module/ncs/aw_archerdig.ncs b/_module/ncs/aw_archerdig.ncs index bf01554..0a8ea06 100644 Binary files a/_module/ncs/aw_archerdig.ncs and b/_module/ncs/aw_archerdig.ncs differ diff --git a/_module/ncs/aw_archermoredig.ncs b/_module/ncs/aw_archermoredig.ncs index c46ab01..adc8446 100644 Binary files a/_module/ncs/aw_archermoredig.ncs and b/_module/ncs/aw_archermoredig.ncs differ diff --git a/_module/ncs/aw_bluespawnin.ncs b/_module/ncs/aw_bluespawnin.ncs index f1dd025..5886d1f 100644 Binary files a/_module/ncs/aw_bluespawnin.ncs and b/_module/ncs/aw_bluespawnin.ncs differ diff --git a/_module/ncs/aw_boogieportal.ncs b/_module/ncs/aw_boogieportal.ncs index 6c529d9..fcce731 100644 Binary files a/_module/ncs/aw_boogieportal.ncs and b/_module/ncs/aw_boogieportal.ncs differ diff --git a/_module/ncs/aw_c_shifter_beh.ncs b/_module/ncs/aw_c_shifter_beh.ncs index 1f9c45b..181bb43 100644 Binary files a/_module/ncs/aw_c_shifter_beh.ncs and b/_module/ncs/aw_c_shifter_beh.ncs differ diff --git a/_module/ncs/aw_c_shifter_dri.ncs b/_module/ncs/aw_c_shifter_dri.ncs index 6d11f20..90abd95 100644 Binary files a/_module/ncs/aw_c_shifter_dri.ncs and b/_module/ncs/aw_c_shifter_dri.ncs differ diff --git a/_module/ncs/aw_c_shifter_min.ncs b/_module/ncs/aw_c_shifter_min.ncs index a000351..8004250 100644 Binary files a/_module/ncs/aw_c_shifter_min.ncs and b/_module/ncs/aw_c_shifter_min.ncs differ diff --git a/_module/ncs/aw_checkdivine.ncs b/_module/ncs/aw_checkdivine.ncs index 1baafde..b305b90 100644 Binary files a/_module/ncs/aw_checkdivine.ncs and b/_module/ncs/aw_checkdivine.ncs differ diff --git a/_module/ncs/aw_checkmagical.ncs b/_module/ncs/aw_checkmagical.ncs index 549931f..6621bee 100644 Binary files a/_module/ncs/aw_checkmagical.ncs and b/_module/ncs/aw_checkmagical.ncs differ diff --git a/_module/ncs/aw_checknegaitve.ncs b/_module/ncs/aw_checknegaitve.ncs index 0fd5e19..af6a2de 100644 Binary files a/_module/ncs/aw_checknegaitve.ncs and b/_module/ncs/aw_checknegaitve.ncs differ diff --git a/_module/ncs/aw_checkpossitiv.ncs b/_module/ncs/aw_checkpossitiv.ncs index 86f12e7..10f6a3e 100644 Binary files a/_module/ncs/aw_checkpossitiv.ncs and b/_module/ncs/aw_checkpossitiv.ncs differ diff --git a/_module/ncs/aw_closetportal.ncs b/_module/ncs/aw_closetportal.ncs index b26d2b3..e0bce94 100644 Binary files a/_module/ncs/aw_closetportal.ncs and b/_module/ncs/aw_closetportal.ncs differ diff --git a/_module/ncs/aw_craft_beholde.ncs b/_module/ncs/aw_craft_beholde.ncs index 21b5613..08033f3 100644 Binary files a/_module/ncs/aw_craft_beholde.ncs and b/_module/ncs/aw_craft_beholde.ncs differ diff --git a/_module/ncs/aw_craft_db.ncs b/_module/ncs/aw_craft_db.ncs index 5e0d8a5..72aa1e9 100644 Binary files a/_module/ncs/aw_craft_db.ncs and b/_module/ncs/aw_craft_db.ncs differ diff --git a/_module/ncs/aw_craft_drider.ncs b/_module/ncs/aw_craft_drider.ncs index 0fc0585..3c4af70 100644 Binary files a/_module/ncs/aw_craft_drider.ncs and b/_module/ncs/aw_craft_drider.ncs differ diff --git a/_module/ncs/aw_craft_dv.ncs b/_module/ncs/aw_craft_dv.ncs index 8118684..29975f3 100644 Binary files a/_module/ncs/aw_craft_dv.ncs and b/_module/ncs/aw_craft_dv.ncs differ diff --git a/_module/ncs/aw_dimdoor.ncs b/_module/ncs/aw_dimdoor.ncs index 39640c4..eb093ff 100644 Binary files a/_module/ncs/aw_dimdoor.ncs and b/_module/ncs/aw_dimdoor.ncs differ diff --git a/_module/ncs/aw_dimdoors.ncs b/_module/ncs/aw_dimdoors.ncs index bac5d23..5dc80e7 100644 Binary files a/_module/ncs/aw_dimdoors.ncs and b/_module/ncs/aw_dimdoors.ncs differ diff --git a/_module/ncs/aw_dire_badger.ncs b/_module/ncs/aw_dire_badger.ncs index 0e9e716..0029c3e 100644 Binary files a/_module/ncs/aw_dire_badger.ncs and b/_module/ncs/aw_dire_badger.ncs differ diff --git a/_module/ncs/aw_dire_bear.ncs b/_module/ncs/aw_dire_bear.ncs index f08c5b2..e000716 100644 Binary files a/_module/ncs/aw_dire_bear.ncs and b/_module/ncs/aw_dire_bear.ncs differ diff --git a/_module/ncs/aw_dire_boar.ncs b/_module/ncs/aw_dire_boar.ncs index 2c497e6..a574028 100644 Binary files a/_module/ncs/aw_dire_boar.ncs and b/_module/ncs/aw_dire_boar.ncs differ diff --git a/_module/ncs/aw_dire_panther.ncs b/_module/ncs/aw_dire_panther.ncs index 117bd7f..d25adf9 100644 Binary files a/_module/ncs/aw_dire_panther.ncs and b/_module/ncs/aw_dire_panther.ncs differ diff --git a/_module/ncs/aw_dire_wolf.ncs b/_module/ncs/aw_dire_wolf.ncs index 493eadb..776a979 100644 Binary files a/_module/ncs/aw_dire_wolf.ncs and b/_module/ncs/aw_dire_wolf.ncs differ diff --git a/_module/ncs/aw_distroydiv.ncs b/_module/ncs/aw_distroydiv.ncs index d63db6c..73b454c 100644 Binary files a/_module/ncs/aw_distroydiv.ncs and b/_module/ncs/aw_distroydiv.ncs differ diff --git a/_module/ncs/aw_distroymag.ncs b/_module/ncs/aw_distroymag.ncs index 8c9040e..8804ae4 100644 Binary files a/_module/ncs/aw_distroymag.ncs and b/_module/ncs/aw_distroymag.ncs differ diff --git a/_module/ncs/aw_distroyneg.ncs b/_module/ncs/aw_distroyneg.ncs index 6b25c74..df0b508 100644 Binary files a/_module/ncs/aw_distroyneg.ncs and b/_module/ncs/aw_distroyneg.ncs differ diff --git a/_module/ncs/aw_distroypos.ncs b/_module/ncs/aw_distroypos.ncs index 5886acd..213efb7 100644 Binary files a/_module/ncs/aw_distroypos.ncs and b/_module/ncs/aw_distroypos.ncs differ diff --git a/_module/ncs/aw_drops.ncs b/_module/ncs/aw_drops.ncs index 24099d6..83b03ff 100644 Binary files a/_module/ncs/aw_drops.ncs and b/_module/ncs/aw_drops.ncs differ diff --git a/_module/ncs/aw_druid_air.ncs b/_module/ncs/aw_druid_air.ncs index 6ef30ca..cad2ea6 100644 Binary files a/_module/ncs/aw_druid_air.ncs and b/_module/ncs/aw_druid_air.ncs differ diff --git a/_module/ncs/aw_druid_badger.ncs b/_module/ncs/aw_druid_badger.ncs index 07b7812..3347040 100644 Binary files a/_module/ncs/aw_druid_badger.ncs and b/_module/ncs/aw_druid_badger.ncs differ diff --git a/_module/ncs/aw_druid_bear.ncs b/_module/ncs/aw_druid_bear.ncs index 394a35d..8fc2531 100644 Binary files a/_module/ncs/aw_druid_bear.ncs and b/_module/ncs/aw_druid_bear.ncs differ diff --git a/_module/ncs/aw_druid_boar.ncs b/_module/ncs/aw_druid_boar.ncs index e750939..a651a44 100644 Binary files a/_module/ncs/aw_druid_boar.ncs and b/_module/ncs/aw_druid_boar.ncs differ diff --git a/_module/ncs/aw_druid_earth.ncs b/_module/ncs/aw_druid_earth.ncs index d319edc..34acbb7 100644 Binary files a/_module/ncs/aw_druid_earth.ncs and b/_module/ncs/aw_druid_earth.ncs differ diff --git a/_module/ncs/aw_druid_fire.ncs b/_module/ncs/aw_druid_fire.ncs index 6b0cb2d..6a1559b 100644 Binary files a/_module/ncs/aw_druid_fire.ncs and b/_module/ncs/aw_druid_fire.ncs differ diff --git a/_module/ncs/aw_druid_lvl12.ncs b/_module/ncs/aw_druid_lvl12.ncs index 1186d9e..a6765c2 100644 Binary files a/_module/ncs/aw_druid_lvl12.ncs and b/_module/ncs/aw_druid_lvl12.ncs differ diff --git a/_module/ncs/aw_druid_lvl16.ncs b/_module/ncs/aw_druid_lvl16.ncs index 6a18ee8..ede9b02 100644 Binary files a/_module/ncs/aw_druid_lvl16.ncs and b/_module/ncs/aw_druid_lvl16.ncs differ diff --git a/_module/ncs/aw_druid_lvl20.ncs b/_module/ncs/aw_druid_lvl20.ncs index 37c59eb..8c97501 100644 Binary files a/_module/ncs/aw_druid_lvl20.ncs and b/_module/ncs/aw_druid_lvl20.ncs differ diff --git a/_module/ncs/aw_druid_lvl5.ncs b/_module/ncs/aw_druid_lvl5.ncs index dc9b14c..b8ef238 100644 Binary files a/_module/ncs/aw_druid_lvl5.ncs and b/_module/ncs/aw_druid_lvl5.ncs differ diff --git a/_module/ncs/aw_druid_panther.ncs b/_module/ncs/aw_druid_panther.ncs index bf811e2..3b6361a 100644 Binary files a/_module/ncs/aw_druid_panther.ncs and b/_module/ncs/aw_druid_panther.ncs differ diff --git a/_module/ncs/aw_druid_water.ncs b/_module/ncs/aw_druid_water.ncs index 58d1e48..4a360b9 100644 Binary files a/_module/ncs/aw_druid_water.ncs and b/_module/ncs/aw_druid_water.ncs differ diff --git a/_module/ncs/aw_druid_wolf.ncs b/_module/ncs/aw_druid_wolf.ncs index 04b4234..937fcb0 100644 Binary files a/_module/ncs/aw_druid_wolf.ncs and b/_module/ncs/aw_druid_wolf.ncs differ diff --git a/_module/ncs/aw_essencecheck.ncs b/_module/ncs/aw_essencecheck.ncs index 5327e96..7683456 100644 Binary files a/_module/ncs/aw_essencecheck.ncs and b/_module/ncs/aw_essencecheck.ncs differ diff --git a/_module/ncs/aw_finalessence.ncs b/_module/ncs/aw_finalessence.ncs index 1d04bfd..3fd5b5a 100644 Binary files a/_module/ncs/aw_finalessence.ncs and b/_module/ncs/aw_finalessence.ncs differ diff --git a/_module/ncs/aw_forcedoor1.ncs b/_module/ncs/aw_forcedoor1.ncs index 6280bd7..d501b2f 100644 Binary files a/_module/ncs/aw_forcedoor1.ncs and b/_module/ncs/aw_forcedoor1.ncs differ diff --git a/_module/ncs/aw_forcedoor2.ncs b/_module/ncs/aw_forcedoor2.ncs index 9bc205b..9c7cfe1 100644 Binary files a/_module/ncs/aw_forcedoor2.ncs and b/_module/ncs/aw_forcedoor2.ncs differ diff --git a/_module/ncs/aw_getshifter.ncs b/_module/ncs/aw_getshifter.ncs index 950d3ac..7ff09d2 100644 Binary files a/_module/ncs/aw_getshifter.ncs and b/_module/ncs/aw_getshifter.ncs differ diff --git a/_module/ncs/aw_getshifterbro.ncs b/_module/ncs/aw_getshifterbro.ncs index 8ce079f..3f04172 100644 Binary files a/_module/ncs/aw_getshifterbro.ncs and b/_module/ncs/aw_getshifterbro.ncs differ diff --git a/_module/ncs/aw_glowing.ncs b/_module/ncs/aw_glowing.ncs index 6471ec2..e4c37f5 100644 Binary files a/_module/ncs/aw_glowing.ncs and b/_module/ncs/aw_glowing.ncs differ diff --git a/_module/ncs/aw_gold1millchec.ncs b/_module/ncs/aw_gold1millchec.ncs index 8928db0..6e5f1d4 100644 Binary files a/_module/ncs/aw_gold1millchec.ncs and b/_module/ncs/aw_gold1millchec.ncs differ diff --git a/_module/ncs/aw_hasbracelet.ncs b/_module/ncs/aw_hasbracelet.ncs index 25441b6..46e1283 100644 Binary files a/_module/ncs/aw_hasbracelet.ncs and b/_module/ncs/aw_hasbracelet.ncs differ diff --git a/_module/ncs/aw_hastooth.ncs b/_module/ncs/aw_hastooth.ncs index 51e1f0c..f341a78 100644 Binary files a/_module/ncs/aw_hastooth.ncs and b/_module/ncs/aw_hastooth.ncs differ diff --git a/_module/ncs/aw_hillsideboim.ncs b/_module/ncs/aw_hillsideboim.ncs index d130a4f..38dada7 100644 Binary files a/_module/ncs/aw_hillsideboim.ncs and b/_module/ncs/aw_hillsideboim.ncs differ diff --git a/_module/ncs/aw_hillsidedant.ncs b/_module/ncs/aw_hillsidedant.ncs index 8798ac1..b022e6b 100644 Binary files a/_module/ncs/aw_hillsidedant.ncs and b/_module/ncs/aw_hillsidedant.ncs differ diff --git a/_module/ncs/aw_hillsideisis.ncs b/_module/ncs/aw_hillsideisis.ncs index 7e12530..48d144e 100644 Binary files a/_module/ncs/aw_hillsideisis.ncs and b/_module/ncs/aw_hillsideisis.ncs differ diff --git a/_module/ncs/aw_hillsideport.ncs b/_module/ncs/aw_hillsideport.ncs index 7b1e425..4524575 100644 Binary files a/_module/ncs/aw_hillsideport.ncs and b/_module/ncs/aw_hillsideport.ncs differ diff --git a/_module/ncs/aw_isisbroken.ncs b/_module/ncs/aw_isisbroken.ncs index 61e6b07..288045a 100644 Binary files a/_module/ncs/aw_isisbroken.ncs and b/_module/ncs/aw_isisbroken.ncs differ diff --git a/_module/ncs/aw_isisditzy.ncs b/_module/ncs/aw_isisditzy.ncs index 68635b5..b579081 100644 Binary files a/_module/ncs/aw_isisditzy.ncs and b/_module/ncs/aw_isisditzy.ncs differ diff --git a/_module/ncs/aw_isisquestion.ncs b/_module/ncs/aw_isisquestion.ncs index dca74c7..e3f170a 100644 Binary files a/_module/ncs/aw_isisquestion.ncs and b/_module/ncs/aw_isisquestion.ncs differ diff --git a/_module/ncs/aw_isistakeessen.ncs b/_module/ncs/aw_isistakeessen.ncs index b58ba84..076c68e 100644 Binary files a/_module/ncs/aw_isistakeessen.ncs and b/_module/ncs/aw_isistakeessen.ncs differ diff --git a/_module/ncs/aw_levelcheck_18.ncs b/_module/ncs/aw_levelcheck_18.ncs index 00e0b4d..d556cbe 100644 Binary files a/_module/ncs/aw_levelcheck_18.ncs and b/_module/ncs/aw_levelcheck_18.ncs differ diff --git a/_module/ncs/aw_moreisis.ncs b/_module/ncs/aw_moreisis.ncs index dca74c7..e3f170a 100644 Binary files a/_module/ncs/aw_moreisis.ncs and b/_module/ncs/aw_moreisis.ncs differ diff --git a/_module/ncs/aw_nightmareente.ncs b/_module/ncs/aw_nightmareente.ncs index 3f161cb..604db06 100644 Binary files a/_module/ncs/aw_nightmareente.ncs and b/_module/ncs/aw_nightmareente.ncs differ diff --git a/_module/ncs/aw_placeablecon.ncs b/_module/ncs/aw_placeablecon.ncs index 3a77cba..864648d 100644 Binary files a/_module/ncs/aw_placeablecon.ncs and b/_module/ncs/aw_placeablecon.ncs differ diff --git a/_module/ncs/aw_portalfromver.ncs b/_module/ncs/aw_portalfromver.ncs index cf926b5..2920f42 100644 Binary files a/_module/ncs/aw_portalfromver.ncs and b/_module/ncs/aw_portalfromver.ncs differ diff --git a/_module/ncs/aw_portaltovera.ncs b/_module/ncs/aw_portaltovera.ncs index 205ef32..8eb458f 100644 Binary files a/_module/ncs/aw_portaltovera.ncs and b/_module/ncs/aw_portaltovera.ncs differ diff --git a/_module/ncs/aw_portaltoveram.ncs b/_module/ncs/aw_portaltoveram.ncs index d82db69..75a8835 100644 Binary files a/_module/ncs/aw_portaltoveram.ncs and b/_module/ncs/aw_portaltoveram.ncs differ diff --git a/_module/ncs/aw_reaperportal.ncs b/_module/ncs/aw_reaperportal.ncs index ab4a10d..c0f7e7a 100644 Binary files a/_module/ncs/aw_reaperportal.ncs and b/_module/ncs/aw_reaperportal.ncs differ diff --git a/_module/ncs/aw_ruinsdoorjump.ncs b/_module/ncs/aw_ruinsdoorjump.ncs index 75057c9..ccb9bdd 100644 Binary files a/_module/ncs/aw_ruinsdoorjump.ncs and b/_module/ncs/aw_ruinsdoorjump.ncs differ diff --git a/_module/ncs/aw_runecheck.ncs b/_module/ncs/aw_runecheck.ncs index abded8e..5dc2a3d 100644 Binary files a/_module/ncs/aw_runecheck.ncs and b/_module/ncs/aw_runecheck.ncs differ diff --git a/_module/ncs/aw_saliacheck.ncs b/_module/ncs/aw_saliacheck.ncs index 61e6b07..288045a 100644 Binary files a/_module/ncs/aw_saliacheck.ncs and b/_module/ncs/aw_saliacheck.ncs differ diff --git a/_module/ncs/aw_saliacheck2.ncs b/_module/ncs/aw_saliacheck2.ncs index d2b4d5f..132d0e5 100644 Binary files a/_module/ncs/aw_saliacheck2.ncs and b/_module/ncs/aw_saliacheck2.ncs differ diff --git a/_module/ncs/aw_shifter_basta.ncs b/_module/ncs/aw_shifter_basta.ncs index d2ead13..ca39695 100644 Binary files a/_module/ncs/aw_shifter_basta.ncs and b/_module/ncs/aw_shifter_basta.ncs differ diff --git a/_module/ncs/aw_shifter_con6.ncs b/_module/ncs/aw_shifter_con6.ncs index 1297b49..9bf61dd 100644 Binary files a/_module/ncs/aw_shifter_con6.ncs and b/_module/ncs/aw_shifter_con6.ncs differ diff --git a/_module/ncs/aw_shifter_cross.ncs b/_module/ncs/aw_shifter_cross.ncs index 16cb149..16e2f51 100644 Binary files a/_module/ncs/aw_shifter_cross.ncs and b/_module/ncs/aw_shifter_cross.ncs differ diff --git a/_module/ncs/aw_shifter_great.ncs b/_module/ncs/aw_shifter_great.ncs index 9ffb97c..fc9e794 100644 Binary files a/_module/ncs/aw_shifter_great.ncs and b/_module/ncs/aw_shifter_great.ncs differ diff --git a/_module/ncs/aw_shifter_lb.ncs b/_module/ncs/aw_shifter_lb.ncs index 5eda303..f5b55cc 100644 Binary files a/_module/ncs/aw_shifter_lb.ncs and b/_module/ncs/aw_shifter_lb.ncs differ diff --git a/_module/ncs/aw_shifter_long.ncs b/_module/ncs/aw_shifter_long.ncs index 5e56cd6..3cf9232 100644 Binary files a/_module/ncs/aw_shifter_long.ncs and b/_module/ncs/aw_shifter_long.ncs differ diff --git a/_module/ncs/aw_shifter_quart.ncs b/_module/ncs/aw_shifter_quart.ncs index 0df3836..b28d2ff 100644 Binary files a/_module/ncs/aw_shifter_quart.ncs and b/_module/ncs/aw_shifter_quart.ncs differ diff --git a/_module/ncs/aw_shifter_sb.ncs b/_module/ncs/aw_shifter_sb.ncs index aa51052..50c53af 100644 Binary files a/_module/ncs/aw_shifter_sb.ncs and b/_module/ncs/aw_shifter_sb.ncs differ diff --git a/_module/ncs/aw_shifter_scyth.ncs b/_module/ncs/aw_shifter_scyth.ncs index 2a94152..c14c360 100644 Binary files a/_module/ncs/aw_shifter_scyth.ncs and b/_module/ncs/aw_shifter_scyth.ncs differ diff --git a/_module/ncs/aw_shifter_short.ncs b/_module/ncs/aw_shifter_short.ncs index fc99e88..3e34839 100644 Binary files a/_module/ncs/aw_shifter_short.ncs and b/_module/ncs/aw_shifter_short.ncs differ diff --git a/_module/ncs/aw_shifter_star.ncs b/_module/ncs/aw_shifter_star.ncs index 8417241..c927c1f 100644 Binary files a/_module/ncs/aw_shifter_star.ncs and b/_module/ncs/aw_shifter_star.ncs differ diff --git a/_module/ncs/aw_shifter_twin.ncs b/_module/ncs/aw_shifter_twin.ncs index 2d2bfe8..618931e 100644 Binary files a/_module/ncs/aw_shifter_twin.ncs and b/_module/ncs/aw_shifter_twin.ncs differ diff --git a/_module/ncs/aw_spell_control.ncs b/_module/ncs/aw_spell_control.ncs index f648e5f..33d55cd 100644 Binary files a/_module/ncs/aw_spell_control.ncs and b/_module/ncs/aw_spell_control.ncs differ diff --git a/_module/ncs/aw_spell_darkven.ncs b/_module/ncs/aw_spell_darkven.ncs index f6018ec..0cee401 100644 Binary files a/_module/ncs/aw_spell_darkven.ncs and b/_module/ncs/aw_spell_darkven.ncs differ diff --git a/_module/ncs/aw_spell_db.ncs b/_module/ncs/aw_spell_db.ncs index 3f380ba..ad89822 100644 Binary files a/_module/ncs/aw_spell_db.ncs and b/_module/ncs/aw_spell_db.ncs differ diff --git a/_module/ncs/aw_startglow.ncs b/_module/ncs/aw_startglow.ncs index 1e40b1e..aea0450 100644 Binary files a/_module/ncs/aw_startglow.ncs and b/_module/ncs/aw_startglow.ncs differ diff --git a/_module/ncs/aw_stopcheater1.ncs b/_module/ncs/aw_stopcheater1.ncs index 1baafde..b305b90 100644 Binary files a/_module/ncs/aw_stopcheater1.ncs and b/_module/ncs/aw_stopcheater1.ncs differ diff --git a/_module/ncs/aw_stopcheater2.ncs b/_module/ncs/aw_stopcheater2.ncs index 2d095f3..cfa9ef8 100644 Binary files a/_module/ncs/aw_stopcheater2.ncs and b/_module/ncs/aw_stopcheater2.ncs differ diff --git a/_module/ncs/aw_take2k.ncs b/_module/ncs/aw_take2k.ncs index 3cf2d98..340b775 100644 Binary files a/_module/ncs/aw_take2k.ncs and b/_module/ncs/aw_take2k.ncs differ diff --git a/_module/ncs/aw_takebracelet.ncs b/_module/ncs/aw_takebracelet.ncs index 13b4b15..c5ca274 100644 Binary files a/_module/ncs/aw_takebracelet.ncs and b/_module/ncs/aw_takebracelet.ncs differ diff --git a/_module/ncs/aw_takerunesforp.ncs b/_module/ncs/aw_takerunesforp.ncs index f10c808..03863de 100644 Binary files a/_module/ncs/aw_takerunesforp.ncs and b/_module/ncs/aw_takerunesforp.ncs differ diff --git a/_module/ncs/aw_tantgold.ncs b/_module/ncs/aw_tantgold.ncs index a2621dd..794f594 100644 Binary files a/_module/ncs/aw_tantgold.ncs and b/_module/ncs/aw_tantgold.ncs differ diff --git a/_module/ncs/aw_totrinity.ncs b/_module/ncs/aw_totrinity.ncs index 884f97c..4117498 100644 Binary files a/_module/ncs/aw_totrinity.ncs and b/_module/ncs/aw_totrinity.ncs differ diff --git a/_module/ncs/aw_whorekeycheck.ncs b/_module/ncs/aw_whorekeycheck.ncs index a1d18e1..70dd5f4 100644 Binary files a/_module/ncs/aw_whorekeycheck.ncs and b/_module/ncs/aw_whorekeycheck.ncs differ diff --git a/_module/ncs/bank_getgold.ncs b/_module/ncs/bank_getgold.ncs index a3f00df..7df05b5 100644 Binary files a/_module/ncs/bank_getgold.ncs and b/_module/ncs/bank_getgold.ncs differ diff --git a/_module/ncs/bank_tellerconvo.ncs b/_module/ncs/bank_tellerconvo.ncs index 25be17c..00ed657 100644 Binary files a/_module/ncs/bank_tellerconvo.ncs and b/_module/ncs/bank_tellerconvo.ncs differ diff --git a/_module/ncs/bank_tellerspawn.ncs b/_module/ncs/bank_tellerspawn.ncs index c69766d..0a6b7f2 100644 Binary files a/_module/ncs/bank_tellerspawn.ncs and b/_module/ncs/bank_tellerspawn.ncs differ diff --git a/_module/ncs/bill_gndr_check.ncs b/_module/ncs/bill_gndr_check.ncs index 322bdea..58d216e 100644 Binary files a/_module/ncs/bill_gndr_check.ncs and b/_module/ncs/bill_gndr_check.ncs differ diff --git a/_module/ncs/bill_key_check.ncs b/_module/ncs/bill_key_check.ncs index 640f9e2..509a5cc 100644 Binary files a/_module/ncs/bill_key_check.ncs and b/_module/ncs/bill_key_check.ncs differ diff --git a/_module/ncs/bill_key_take.ncs b/_module/ncs/bill_key_take.ncs index b6b98d3..a9f391e 100644 Binary files a/_module/ncs/bill_key_take.ncs and b/_module/ncs/bill_key_take.ncs differ diff --git a/_module/ncs/bill_var_check.ncs b/_module/ncs/bill_var_check.ncs index 19ce00e..faf1bcb 100644 Binary files a/_module/ncs/bill_var_check.ncs and b/_module/ncs/bill_var_check.ncs differ diff --git a/_module/ncs/btr_chest_reset.ncs b/_module/ncs/btr_chest_reset.ncs index 31102a3..702cf8e 100644 Binary files a/_module/ncs/btr_chest_reset.ncs and b/_module/ncs/btr_chest_reset.ncs differ diff --git a/_module/ncs/bush_conv_start.ncs b/_module/ncs/bush_conv_start.ncs index 2b84b9d..8af42c3 100644 Binary files a/_module/ncs/bush_conv_start.ncs and b/_module/ncs/bush_conv_start.ncs differ diff --git a/_module/ncs/bvcapt_to_trin.ncs b/_module/ncs/bvcapt_to_trin.ncs index f6cb292..5bc6b07 100644 Binary files a/_module/ncs/bvcapt_to_trin.ncs and b/_module/ncs/bvcapt_to_trin.ncs differ diff --git a/_module/ncs/bvcaptain_got20k.ncs b/_module/ncs/bvcaptain_got20k.ncs index 06c36fd..957f9b5 100644 Binary files a/_module/ncs/bvcaptain_got20k.ncs and b/_module/ncs/bvcaptain_got20k.ncs differ diff --git a/_module/ncs/bvcaptain_levchk.ncs b/_module/ncs/bvcaptain_levchk.ncs index 80e015c..48ed2af 100644 Binary files a/_module/ncs/bvcaptain_levchk.ncs and b/_module/ncs/bvcaptain_levchk.ncs differ diff --git a/_module/ncs/bvcaptain_port.ncs b/_module/ncs/bvcaptain_port.ncs index 8c2b61e..8944329 100644 Binary files a/_module/ncs/bvcaptain_port.ncs and b/_module/ncs/bvcaptain_port.ncs differ diff --git a/_module/ncs/cb_11port_01.ncs b/_module/ncs/cb_11port_01.ncs index d3aede9..c89a5b0 100644 Binary files a/_module/ncs/cb_11port_01.ncs and b/_module/ncs/cb_11port_01.ncs differ diff --git a/_module/ncs/cb_2ndmonkjump.ncs b/_module/ncs/cb_2ndmonkjump.ncs index 1dcd8f0..d7a55eb 100644 Binary files a/_module/ncs/cb_2ndmonkjump.ncs and b/_module/ncs/cb_2ndmonkjump.ncs differ diff --git a/_module/ncs/cb_areacold.ncs b/_module/ncs/cb_areacold.ncs index 62b3f86..6bd28c1 100644 Binary files a/_module/ncs/cb_areacold.ncs and b/_module/ncs/cb_areacold.ncs differ diff --git a/_module/ncs/cb_asyglobe.ncs b/_module/ncs/cb_asyglobe.ncs index 8496fd3..29f555a 100644 Binary files a/_module/ncs/cb_asyglobe.ncs and b/_module/ncs/cb_asyglobe.ncs differ diff --git a/_module/ncs/cb_atl_port.ncs b/_module/ncs/cb_atl_port.ncs index 3a3ea43..2df0360 100644 Binary files a/_module/ncs/cb_atl_port.ncs and b/_module/ncs/cb_atl_port.ncs differ diff --git a/_module/ncs/cb_atl_take.ncs b/_module/ncs/cb_atl_take.ncs index 265574a..63ef61b 100644 Binary files a/_module/ncs/cb_atl_take.ncs and b/_module/ncs/cb_atl_take.ncs differ diff --git a/_module/ncs/cb_attack_all.ncs b/_module/ncs/cb_attack_all.ncs index 7509b98..5288781 100644 Binary files a/_module/ncs/cb_attack_all.ncs and b/_module/ncs/cb_attack_all.ncs differ diff --git a/_module/ncs/cb_attack_pc.ncs b/_module/ncs/cb_attack_pc.ncs index 3ae3efc..116ea36 100644 Binary files a/_module/ncs/cb_attack_pc.ncs and b/_module/ncs/cb_attack_pc.ncs differ diff --git a/_module/ncs/cb_battlefield.ncs b/_module/ncs/cb_battlefield.ncs index f7f6e10..53e9a7c 100644 Binary files a/_module/ncs/cb_battlefield.ncs and b/_module/ncs/cb_battlefield.ncs differ diff --git a/_module/ncs/cb_boneydie.ncs b/_module/ncs/cb_boneydie.ncs index 06eef33..20f0642 100644 Binary files a/_module/ncs/cb_boneydie.ncs and b/_module/ncs/cb_boneydie.ncs differ diff --git a/_module/ncs/cb_brewreward.ncs b/_module/ncs/cb_brewreward.ncs index 4b50ea7..a0dac95 100644 Binary files a/_module/ncs/cb_brewreward.ncs and b/_module/ncs/cb_brewreward.ncs differ diff --git a/_module/ncs/cb_caliman2.ncs b/_module/ncs/cb_caliman2.ncs index 03564b8..765ec5b 100644 Binary files a/_module/ncs/cb_caliman2.ncs and b/_module/ncs/cb_caliman2.ncs differ diff --git a/_module/ncs/cb_chk_100k.ncs b/_module/ncs/cb_chk_100k.ncs index 7b16c12..f9ca939 100644 Binary files a/_module/ncs/cb_chk_100k.ncs and b/_module/ncs/cb_chk_100k.ncs differ diff --git a/_module/ncs/cb_chk_25k.ncs b/_module/ncs/cb_chk_25k.ncs index ba4e9dd..52a53f7 100644 Binary files a/_module/ncs/cb_chk_25k.ncs and b/_module/ncs/cb_chk_25k.ncs differ diff --git a/_module/ncs/cb_chk_2k.ncs b/_module/ncs/cb_chk_2k.ncs index 5e13915..6a9c1cb 100644 Binary files a/_module/ncs/cb_chk_2k.ncs and b/_module/ncs/cb_chk_2k.ncs differ diff --git a/_module/ncs/cb_chk_30.ncs b/_module/ncs/cb_chk_30.ncs index 3343ef5..272b4f8 100644 Binary files a/_module/ncs/cb_chk_30.ncs and b/_module/ncs/cb_chk_30.ncs differ diff --git a/_module/ncs/cb_chk_5k.ncs b/_module/ncs/cb_chk_5k.ncs index b16551a..788aa5e 100644 Binary files a/_module/ncs/cb_chk_5k.ncs and b/_module/ncs/cb_chk_5k.ncs differ diff --git a/_module/ncs/cb_chk_acidstick.ncs b/_module/ncs/cb_chk_acidstick.ncs index 6269c58..32e9d99 100644 Binary files a/_module/ncs/cb_chk_acidstick.ncs and b/_module/ncs/cb_chk_acidstick.ncs differ diff --git a/_module/ncs/cb_chk_atlstone.ncs b/_module/ncs/cb_chk_atlstone.ncs index 7ae040c..ef568df 100644 Binary files a/_module/ncs/cb_chk_atlstone.ncs and b/_module/ncs/cb_chk_atlstone.ncs differ diff --git a/_module/ncs/cb_chk_blueplan.ncs b/_module/ncs/cb_chk_blueplan.ncs index 486316e..9796814 100644 Binary files a/_module/ncs/cb_chk_blueplan.ncs and b/_module/ncs/cb_chk_blueplan.ncs differ diff --git a/_module/ncs/cb_chk_booknoble.ncs b/_module/ncs/cb_chk_booknoble.ncs index 9f3c3c1..4206d6c 100644 Binary files a/_module/ncs/cb_chk_booknoble.ncs and b/_module/ncs/cb_chk_booknoble.ncs differ diff --git a/_module/ncs/cb_chk_dragammy.ncs b/_module/ncs/cb_chk_dragammy.ncs index 3e99d7c..4518b14 100644 Binary files a/_module/ncs/cb_chk_dragammy.ncs and b/_module/ncs/cb_chk_dragammy.ncs differ diff --git a/_module/ncs/cb_chk_drageverg.ncs b/_module/ncs/cb_chk_drageverg.ncs index 95594e4..9298d13 100644 Binary files a/_module/ncs/cb_chk_drageverg.ncs and b/_module/ncs/cb_chk_drageverg.ncs differ diff --git a/_module/ncs/cb_chk_drownoble.ncs b/_module/ncs/cb_chk_drownoble.ncs index c9bb933..ace887f 100644 Binary files a/_module/ncs/cb_chk_drownoble.ncs and b/_module/ncs/cb_chk_drownoble.ncs differ diff --git a/_module/ncs/cb_chk_evilorb.ncs b/_module/ncs/cb_chk_evilorb.ncs index 5593d6f..7f6f852 100644 Binary files a/_module/ncs/cb_chk_evilorb.ncs and b/_module/ncs/cb_chk_evilorb.ncs differ diff --git a/_module/ncs/cb_chk_globe.ncs b/_module/ncs/cb_chk_globe.ncs index ecb1c72..e63d92f 100644 Binary files a/_module/ncs/cb_chk_globe.ncs and b/_module/ncs/cb_chk_globe.ncs differ diff --git a/_module/ncs/cb_chk_goodorb.ncs b/_module/ncs/cb_chk_goodorb.ncs index df5e751..465503f 100644 Binary files a/_module/ncs/cb_chk_goodorb.ncs and b/_module/ncs/cb_chk_goodorb.ncs differ diff --git a/_module/ncs/cb_chk_grail.ncs b/_module/ncs/cb_chk_grail.ncs index 2d20a76..69b4ecc 100644 Binary files a/_module/ncs/cb_chk_grail.ncs and b/_module/ncs/cb_chk_grail.ncs differ diff --git a/_module/ncs/cb_chk_greenplan.ncs b/_module/ncs/cb_chk_greenplan.ncs index 92d06d5..deaf987 100644 Binary files a/_module/ncs/cb_chk_greenplan.ncs and b/_module/ncs/cb_chk_greenplan.ncs differ diff --git a/_module/ncs/cb_chk_grial.ncs b/_module/ncs/cb_chk_grial.ncs index 9556eac..9ebc2d4 100644 Binary files a/_module/ncs/cb_chk_grial.ncs and b/_module/ncs/cb_chk_grial.ncs differ diff --git a/_module/ncs/cb_chk_horn.ncs b/_module/ncs/cb_chk_horn.ncs index 34a2cc8..dcbe1e7 100644 Binary files a/_module/ncs/cb_chk_horn.ncs and b/_module/ncs/cb_chk_horn.ncs differ diff --git a/_module/ncs/cb_chk_lith.ncs b/_module/ncs/cb_chk_lith.ncs index 7aebfdd..5b640c0 100644 Binary files a/_module/ncs/cb_chk_lith.ncs and b/_module/ncs/cb_chk_lith.ncs differ diff --git a/_module/ncs/cb_chk_lstaff.ncs b/_module/ncs/cb_chk_lstaff.ncs index 2ea7499..cddcc48 100644 Binary files a/_module/ncs/cb_chk_lstaff.ncs and b/_module/ncs/cb_chk_lstaff.ncs differ diff --git a/_module/ncs/cb_chk_mahead.ncs b/_module/ncs/cb_chk_mahead.ncs index 1c89d81..83dab05 100644 Binary files a/_module/ncs/cb_chk_mahead.ncs and b/_module/ncs/cb_chk_mahead.ncs differ diff --git a/_module/ncs/cb_chk_mug.ncs b/_module/ncs/cb_chk_mug.ncs index 025bc9c..d5e3edd 100644 Binary files a/_module/ncs/cb_chk_mug.ncs and b/_module/ncs/cb_chk_mug.ncs differ diff --git a/_module/ncs/cb_chk_redplan.ncs b/_module/ncs/cb_chk_redplan.ncs index 60f5bf0..3eabdb2 100644 Binary files a/_module/ncs/cb_chk_redplan.ncs and b/_module/ncs/cb_chk_redplan.ncs differ diff --git a/_module/ncs/cb_chk_scrollals.ncs b/_module/ncs/cb_chk_scrollals.ncs index 915ddb9..ccfa6d3 100644 Binary files a/_module/ncs/cb_chk_scrollals.ncs and b/_module/ncs/cb_chk_scrollals.ncs differ diff --git a/_module/ncs/cb_chk_scrolls.ncs b/_module/ncs/cb_chk_scrolls.ncs index 914bc34..72832bd 100644 Binary files a/_module/ncs/cb_chk_scrolls.ncs and b/_module/ncs/cb_chk_scrolls.ncs differ diff --git a/_module/ncs/cb_chk_serum.ncs b/_module/ncs/cb_chk_serum.ncs index 6731ba2..e71030c 100644 Binary files a/_module/ncs/cb_chk_serum.ncs and b/_module/ncs/cb_chk_serum.ncs differ diff --git a/_module/ncs/cb_chk_taghead.ncs b/_module/ncs/cb_chk_taghead.ncs index 999a8a6..df1cc59 100644 Binary files a/_module/ncs/cb_chk_taghead.ncs and b/_module/ncs/cb_chk_taghead.ncs differ diff --git a/_module/ncs/cb_chk_tmbattle.ncs b/_module/ncs/cb_chk_tmbattle.ncs index 1044eac..f8376c2 100644 Binary files a/_module/ncs/cb_chk_tmbattle.ncs and b/_module/ncs/cb_chk_tmbattle.ncs differ diff --git a/_module/ncs/cb_chk_yoga.ncs b/_module/ncs/cb_chk_yoga.ncs index f0f274e..f6c9869 100644 Binary files a/_module/ncs/cb_chk_yoga.ncs and b/_module/ncs/cb_chk_yoga.ncs differ diff --git a/_module/ncs/cb_chklvl1.ncs b/_module/ncs/cb_chklvl1.ncs index 4e8fbd9..8bd0771 100644 Binary files a/_module/ncs/cb_chklvl1.ncs and b/_module/ncs/cb_chklvl1.ncs differ diff --git a/_module/ncs/cb_cloakpotion.ncs b/_module/ncs/cb_cloakpotion.ncs index 9b43629..a9a7f5b 100644 Binary files a/_module/ncs/cb_cloakpotion.ncs and b/_module/ncs/cb_cloakpotion.ncs differ diff --git a/_module/ncs/cb_cloakwall.ncs b/_module/ncs/cb_cloakwall.ncs index c1d267e..af45d17 100644 Binary files a/_module/ncs/cb_cloakwall.ncs and b/_module/ncs/cb_cloakwall.ncs differ diff --git a/_module/ncs/cb_conv_placable.ncs b/_module/ncs/cb_conv_placable.ncs index 3a77cba..864648d 100644 Binary files a/_module/ncs/cb_conv_placable.ncs and b/_module/ncs/cb_conv_placable.ncs differ diff --git a/_module/ncs/cb_cushions1.ncs b/_module/ncs/cb_cushions1.ncs index 02e8b3b..db87f58 100644 Binary files a/_module/ncs/cb_cushions1.ncs and b/_module/ncs/cb_cushions1.ncs differ diff --git a/_module/ncs/cb_dirthboom.ncs b/_module/ncs/cb_dirthboom.ncs index dca9074..4f4f79e 100644 Binary files a/_module/ncs/cb_dirthboom.ncs and b/_module/ncs/cb_dirthboom.ncs differ diff --git a/_module/ncs/cb_dirthport1.ncs b/_module/ncs/cb_dirthport1.ncs index 1293ab9..5945281 100644 Binary files a/_module/ncs/cb_dirthport1.ncs and b/_module/ncs/cb_dirthport1.ncs differ diff --git a/_module/ncs/cb_dirthport10.ncs b/_module/ncs/cb_dirthport10.ncs index 8cbc4c4..dfea10e 100644 Binary files a/_module/ncs/cb_dirthport10.ncs and b/_module/ncs/cb_dirthport10.ncs differ diff --git a/_module/ncs/cb_dirthport11.ncs b/_module/ncs/cb_dirthport11.ncs index 705b6c9..f46324b 100644 Binary files a/_module/ncs/cb_dirthport11.ncs and b/_module/ncs/cb_dirthport11.ncs differ diff --git a/_module/ncs/cb_dirthport2.ncs b/_module/ncs/cb_dirthport2.ncs index f199c40..a3a285e 100644 Binary files a/_module/ncs/cb_dirthport2.ncs and b/_module/ncs/cb_dirthport2.ncs differ diff --git a/_module/ncs/cb_dirthport3.ncs b/_module/ncs/cb_dirthport3.ncs index e341e7b..056dca2 100644 Binary files a/_module/ncs/cb_dirthport3.ncs and b/_module/ncs/cb_dirthport3.ncs differ diff --git a/_module/ncs/cb_dirthport4.ncs b/_module/ncs/cb_dirthport4.ncs index 7220fab..52446d2 100644 Binary files a/_module/ncs/cb_dirthport4.ncs and b/_module/ncs/cb_dirthport4.ncs differ diff --git a/_module/ncs/cb_dirthport5.ncs b/_module/ncs/cb_dirthport5.ncs index 5840bbe..96dea16 100644 Binary files a/_module/ncs/cb_dirthport5.ncs and b/_module/ncs/cb_dirthport5.ncs differ diff --git a/_module/ncs/cb_dirthport6.ncs b/_module/ncs/cb_dirthport6.ncs index 274dab0..c0e8693 100644 Binary files a/_module/ncs/cb_dirthport6.ncs and b/_module/ncs/cb_dirthport6.ncs differ diff --git a/_module/ncs/cb_dirthport7.ncs b/_module/ncs/cb_dirthport7.ncs index 1681fda..e7481ab 100644 Binary files a/_module/ncs/cb_dirthport7.ncs and b/_module/ncs/cb_dirthport7.ncs differ diff --git a/_module/ncs/cb_dirthport8.ncs b/_module/ncs/cb_dirthport8.ncs index 9b9b271..1711ee0 100644 Binary files a/_module/ncs/cb_dirthport8.ncs and b/_module/ncs/cb_dirthport8.ncs differ diff --git a/_module/ncs/cb_dirthport9.ncs b/_module/ncs/cb_dirthport9.ncs index c6df18b..464965c 100644 Binary files a/_module/ncs/cb_dirthport9.ncs and b/_module/ncs/cb_dirthport9.ncs differ diff --git a/_module/ncs/cb_disarmtrap.ncs b/_module/ncs/cb_disarmtrap.ncs index f018fef..8b6b1b4 100644 Binary files a/_module/ncs/cb_disarmtrap.ncs and b/_module/ncs/cb_disarmtrap.ncs differ diff --git a/_module/ncs/cb_dmstore2.ncs b/_module/ncs/cb_dmstore2.ncs index d740d52..3431cc0 100644 Binary files a/_module/ncs/cb_dmstore2.ncs and b/_module/ncs/cb_dmstore2.ncs differ diff --git a/_module/ncs/cb_dontattacknpc.ncs b/_module/ncs/cb_dontattacknpc.ncs index 9732b4a..8276bd0 100644 Binary files a/_module/ncs/cb_dontattacknpc.ncs and b/_module/ncs/cb_dontattacknpc.ncs differ diff --git a/_module/ncs/cb_doorspeak.ncs b/_module/ncs/cb_doorspeak.ncs index 208e0cf..a067f78 100644 Binary files a/_module/ncs/cb_doorspeak.ncs and b/_module/ncs/cb_doorspeak.ncs differ diff --git a/_module/ncs/cb_doorspeak1.ncs b/_module/ncs/cb_doorspeak1.ncs index 59dea1d..ddc236d 100644 Binary files a/_module/ncs/cb_doorspeak1.ncs and b/_module/ncs/cb_doorspeak1.ncs differ diff --git a/_module/ncs/cb_doorspeak2.ncs b/_module/ncs/cb_doorspeak2.ncs index a86b9b5..e8f22db 100644 Binary files a/_module/ncs/cb_doorspeak2.ncs and b/_module/ncs/cb_doorspeak2.ncs differ diff --git a/_module/ncs/cb_doorspeak3.ncs b/_module/ncs/cb_doorspeak3.ncs index 6da2b2b..e5a0035 100644 Binary files a/_module/ncs/cb_doorspeak3.ncs and b/_module/ncs/cb_doorspeak3.ncs differ diff --git a/_module/ncs/cb_drag_backup.ncs b/_module/ncs/cb_drag_backup.ncs index 20c8ab7..1886a49 100644 Binary files a/_module/ncs/cb_drag_backup.ncs and b/_module/ncs/cb_drag_backup.ncs differ diff --git a/_module/ncs/cb_dragon_light.ncs b/_module/ncs/cb_dragon_light.ncs index 10941b4..424b072 100644 Binary files a/_module/ncs/cb_dragon_light.ncs and b/_module/ncs/cb_dragon_light.ncs differ diff --git a/_module/ncs/cb_drowattack.ncs b/_module/ncs/cb_drowattack.ncs index 856f7d7..c2d835f 100644 Binary files a/_module/ncs/cb_drowattack.ncs and b/_module/ncs/cb_drowattack.ncs differ diff --git a/_module/ncs/cb_elfattack1.ncs b/_module/ncs/cb_elfattack1.ncs index 0d311cc..4738e69 100644 Binary files a/_module/ncs/cb_elfattack1.ncs and b/_module/ncs/cb_elfattack1.ncs differ diff --git a/_module/ncs/cb_elfattack2.ncs b/_module/ncs/cb_elfattack2.ncs index 1d70651..ecf79e9 100644 Binary files a/_module/ncs/cb_elfattack2.ncs and b/_module/ncs/cb_elfattack2.ncs differ diff --git a/_module/ncs/cb_elfattack3.ncs b/_module/ncs/cb_elfattack3.ncs index 4e05f85..f16a88c 100644 Binary files a/_module/ncs/cb_elfattack3.ncs and b/_module/ncs/cb_elfattack3.ncs differ diff --git a/_module/ncs/cb_elfmerch.ncs b/_module/ncs/cb_elfmerch.ncs index 67daf7e..aa0b634 100644 Binary files a/_module/ncs/cb_elfmerch.ncs and b/_module/ncs/cb_elfmerch.ncs differ diff --git a/_module/ncs/cb_elvendar_ente.ncs b/_module/ncs/cb_elvendar_ente.ncs index 56198af..618221b 100644 Binary files a/_module/ncs/cb_elvendar_ente.ncs and b/_module/ncs/cb_elvendar_ente.ncs differ diff --git a/_module/ncs/cb_energydestroy.ncs b/_module/ncs/cb_energydestroy.ncs index 4962d14..1691414 100644 Binary files a/_module/ncs/cb_energydestroy.ncs and b/_module/ncs/cb_energydestroy.ncs differ diff --git a/_module/ncs/cb_enter_battle.ncs b/_module/ncs/cb_enter_battle.ncs index efc4eec..d0d0852 100644 Binary files a/_module/ncs/cb_enter_battle.ncs and b/_module/ncs/cb_enter_battle.ncs differ diff --git a/_module/ncs/cb_equipmelee.ncs b/_module/ncs/cb_equipmelee.ncs index 933cf77..c1faa20 100644 Binary files a/_module/ncs/cb_equipmelee.ncs and b/_module/ncs/cb_equipmelee.ncs differ diff --git a/_module/ncs/cb_evil.ncs b/_module/ncs/cb_evil.ncs index 707b29c..9b15003 100644 Binary files a/_module/ncs/cb_evil.ncs and b/_module/ncs/cb_evil.ncs differ diff --git a/_module/ncs/cb_evilport20.ncs b/_module/ncs/cb_evilport20.ncs index df9f9c8..7144e9b 100644 Binary files a/_module/ncs/cb_evilport20.ncs and b/_module/ncs/cb_evilport20.ncs differ diff --git a/_module/ncs/cb_forcedoor1.ncs b/_module/ncs/cb_forcedoor1.ncs index dded076..be93fa9 100644 Binary files a/_module/ncs/cb_forcedoor1.ncs and b/_module/ncs/cb_forcedoor1.ncs differ diff --git a/_module/ncs/cb_forcedoor2.ncs b/_module/ncs/cb_forcedoor2.ncs index 1883840..bb9736e 100644 Binary files a/_module/ncs/cb_forcedoor2.ncs and b/_module/ncs/cb_forcedoor2.ncs differ diff --git a/_module/ncs/cb_forcedoor3.ncs b/_module/ncs/cb_forcedoor3.ncs index aa75763..c1c79b6 100644 Binary files a/_module/ncs/cb_forcedoor3.ncs and b/_module/ncs/cb_forcedoor3.ncs differ diff --git a/_module/ncs/cb_giv_50k.ncs b/_module/ncs/cb_giv_50k.ncs index 6a9434e..39460ba 100644 Binary files a/_module/ncs/cb_giv_50k.ncs and b/_module/ncs/cb_giv_50k.ncs differ diff --git a/_module/ncs/cb_giv_draggem.ncs b/_module/ncs/cb_giv_draggem.ncs index e74599e..a4443c9 100644 Binary files a/_module/ncs/cb_giv_draggem.ncs and b/_module/ncs/cb_giv_draggem.ncs differ diff --git a/_module/ncs/cb_givebow.ncs b/_module/ncs/cb_givebow.ncs index 92d7516..475207a 100644 Binary files a/_module/ncs/cb_givebow.ncs and b/_module/ncs/cb_givebow.ncs differ diff --git a/_module/ncs/cb_givehorn.ncs b/_module/ncs/cb_givehorn.ncs index 23d781b..0ce138a 100644 Binary files a/_module/ncs/cb_givehorn.ncs and b/_module/ncs/cb_givehorn.ncs differ diff --git a/_module/ncs/cb_good.ncs b/_module/ncs/cb_good.ncs index e7c86b4..859241b 100644 Binary files a/_module/ncs/cb_good.ncs and b/_module/ncs/cb_good.ncs differ diff --git a/_module/ncs/cb_goodport20.ncs b/_module/ncs/cb_goodport20.ncs index fd0add1..8b3b443 100644 Binary files a/_module/ncs/cb_goodport20.ncs and b/_module/ncs/cb_goodport20.ncs differ diff --git a/_module/ncs/cb_gotojail.ncs b/_module/ncs/cb_gotojail.ncs index 1d883c4..4d46119 100644 Binary files a/_module/ncs/cb_gotojail.ncs and b/_module/ncs/cb_gotojail.ncs differ diff --git a/_module/ncs/cb_gve_atlstone.ncs b/_module/ncs/cb_gve_atlstone.ncs index db2a898..0ba1940 100644 Binary files a/_module/ncs/cb_gve_atlstone.ncs and b/_module/ncs/cb_gve_atlstone.ncs differ diff --git a/_module/ncs/cb_janopport.ncs b/_module/ncs/cb_janopport.ncs index 2d9c96f..87320ad 100644 Binary files a/_module/ncs/cb_janopport.ncs and b/_module/ncs/cb_janopport.ncs differ diff --git a/_module/ncs/cb_jewl_tool.ncs b/_module/ncs/cb_jewl_tool.ncs index a9e307c..0958ec3 100644 Binary files a/_module/ncs/cb_jewl_tool.ncs and b/_module/ncs/cb_jewl_tool.ncs differ diff --git a/_module/ncs/cb_jez_death.ncs b/_module/ncs/cb_jez_death.ncs index 25d95d7..472265c 100644 Binary files a/_module/ncs/cb_jez_death.ncs and b/_module/ncs/cb_jez_death.ncs differ diff --git a/_module/ncs/cb_klaumdoor.ncs b/_module/ncs/cb_klaumdoor.ncs index 36f8a59..7a353c9 100644 Binary files a/_module/ncs/cb_klaumdoor.ncs and b/_module/ncs/cb_klaumdoor.ncs differ diff --git a/_module/ncs/cb_klaumspawn1.ncs b/_module/ncs/cb_klaumspawn1.ncs index 95802da..4e74441 100644 Binary files a/_module/ncs/cb_klaumspawn1.ncs and b/_module/ncs/cb_klaumspawn1.ncs differ diff --git a/_module/ncs/cb_legionport1.ncs b/_module/ncs/cb_legionport1.ncs index 18105e9..cdbcb22 100644 Binary files a/_module/ncs/cb_legionport1.ncs and b/_module/ncs/cb_legionport1.ncs differ diff --git a/_module/ncs/cb_lithport1.ncs b/_module/ncs/cb_lithport1.ncs index 8707754..13aa632 100644 Binary files a/_module/ncs/cb_lithport1.ncs and b/_module/ncs/cb_lithport1.ncs differ diff --git a/_module/ncs/cb_lithport2.ncs b/_module/ncs/cb_lithport2.ncs index 069b008..e76fb10 100644 Binary files a/_module/ncs/cb_lithport2.ncs and b/_module/ncs/cb_lithport2.ncs differ diff --git a/_module/ncs/cb_lithport3.ncs b/_module/ncs/cb_lithport3.ncs index 069b008..e76fb10 100644 Binary files a/_module/ncs/cb_lithport3.ncs and b/_module/ncs/cb_lithport3.ncs differ diff --git a/_module/ncs/cb_lithporttab.ncs b/_module/ncs/cb_lithporttab.ncs index cde101c..3513906 100644 Binary files a/_module/ncs/cb_lithporttab.ncs and b/_module/ncs/cb_lithporttab.ncs differ diff --git a/_module/ncs/cb_malnipstore.ncs b/_module/ncs/cb_malnipstore.ncs index 837b8e4..5119e4e 100644 Binary files a/_module/ncs/cb_malnipstore.ncs and b/_module/ncs/cb_malnipstore.ncs differ diff --git a/_module/ncs/cb_max_lvl5.ncs b/_module/ncs/cb_max_lvl5.ncs index b386e23..942ec92 100644 Binary files a/_module/ncs/cb_max_lvl5.ncs and b/_module/ncs/cb_max_lvl5.ncs differ diff --git a/_module/ncs/cb_max_lvl8.ncs b/_module/ncs/cb_max_lvl8.ncs index b00e5f9..f248b65 100644 Binary files a/_module/ncs/cb_max_lvl8.ncs and b/_module/ncs/cb_max_lvl8.ncs differ diff --git a/_module/ncs/cb_meditate.ncs b/_module/ncs/cb_meditate.ncs index 51b66e9..4ef4907 100644 Binary files a/_module/ncs/cb_meditate.ncs and b/_module/ncs/cb_meditate.ncs differ diff --git a/_module/ncs/cb_meditate_per.ncs b/_module/ncs/cb_meditate_per.ncs index cb01ef1..c017ecf 100644 Binary files a/_module/ncs/cb_meditate_per.ncs and b/_module/ncs/cb_meditate_per.ncs differ diff --git a/_module/ncs/cb_minddeath.ncs b/_module/ncs/cb_minddeath.ncs index cc2c66b..b073679 100644 Binary files a/_module/ncs/cb_minddeath.ncs and b/_module/ncs/cb_minddeath.ncs differ diff --git a/_module/ncs/cb_monkattack.ncs b/_module/ncs/cb_monkattack.ncs index 5d5996d..24440b0 100644 Binary files a/_module/ncs/cb_monkattack.ncs and b/_module/ncs/cb_monkattack.ncs differ diff --git a/_module/ncs/cb_monkworship.ncs b/_module/ncs/cb_monkworship.ncs index 9638a5f..c49d8dd 100644 Binary files a/_module/ncs/cb_monkworship.ncs and b/_module/ncs/cb_monkworship.ncs differ diff --git a/_module/ncs/cb_neutral.ncs b/_module/ncs/cb_neutral.ncs index c2ec378..3ef732f 100644 Binary files a/_module/ncs/cb_neutral.ncs and b/_module/ncs/cb_neutral.ncs differ diff --git a/_module/ncs/cb_newport.ncs b/_module/ncs/cb_newport.ncs index bdeeb54..6eb72e1 100644 Binary files a/_module/ncs/cb_newport.ncs and b/_module/ncs/cb_newport.ncs differ diff --git a/_module/ncs/cb_npc_bash.ncs b/_module/ncs/cb_npc_bash.ncs index 0b3dbdd..c16259a 100644 Binary files a/_module/ncs/cb_npc_bash.ncs and b/_module/ncs/cb_npc_bash.ncs differ diff --git a/_module/ncs/cb_npc_chairsit.ncs b/_module/ncs/cb_npc_chairsit.ncs index 515f1af..c3eb154 100644 Binary files a/_module/ncs/cb_npc_chairsit.ncs and b/_module/ncs/cb_npc_chairsit.ncs differ diff --git a/_module/ncs/cb_npc_facedown.ncs b/_module/ncs/cb_npc_facedown.ncs index 56e2b7c..11ced57 100644 Binary files a/_module/ncs/cb_npc_facedown.ncs and b/_module/ncs/cb_npc_facedown.ncs differ diff --git a/_module/ncs/cb_npc_meditate.ncs b/_module/ncs/cb_npc_meditate.ncs index c20762e..b0c87ab 100644 Binary files a/_module/ncs/cb_npc_meditate.ncs and b/_module/ncs/cb_npc_meditate.ncs differ diff --git a/_module/ncs/cb_npc_plead.ncs b/_module/ncs/cb_npc_plead.ncs index 35b46e7..e298034 100644 Binary files a/_module/ncs/cb_npc_plead.ncs and b/_module/ncs/cb_npc_plead.ncs differ diff --git a/_module/ncs/cb_npc_read.ncs b/_module/ncs/cb_npc_read.ncs index e29f428..1faf024 100644 Binary files a/_module/ncs/cb_npc_read.ncs and b/_module/ncs/cb_npc_read.ncs differ diff --git a/_module/ncs/cb_npc_read2.ncs b/_module/ncs/cb_npc_read2.ncs index bbae25f..0de4ae3 100644 Binary files a/_module/ncs/cb_npc_read2.ncs and b/_module/ncs/cb_npc_read2.ncs differ diff --git a/_module/ncs/cb_npc_saulte.ncs b/_module/ncs/cb_npc_saulte.ncs index bb671d2..2efc1b1 100644 Binary files a/_module/ncs/cb_npc_saulte.ncs and b/_module/ncs/cb_npc_saulte.ncs differ diff --git a/_module/ncs/cb_npc_sitdown.ncs b/_module/ncs/cb_npc_sitdown.ncs index c7a9bfb..82db3ab 100644 Binary files a/_module/ncs/cb_npc_sitdown.ncs and b/_module/ncs/cb_npc_sitdown.ncs differ diff --git a/_module/ncs/cb_npc_sleep_hb.ncs b/_module/ncs/cb_npc_sleep_hb.ncs index f71ad32..65d9ab4 100644 Binary files a/_module/ncs/cb_npc_sleep_hb.ncs and b/_module/ncs/cb_npc_sleep_hb.ncs differ diff --git a/_module/ncs/cb_npc_talkforce.ncs b/_module/ncs/cb_npc_talkforce.ncs index 04871fd..3c590f0 100644 Binary files a/_module/ncs/cb_npc_talkforce.ncs and b/_module/ncs/cb_npc_talkforce.ncs differ diff --git a/_module/ncs/cb_npcmeditat_hb.ncs b/_module/ncs/cb_npcmeditat_hb.ncs index 22ddcaf..8466f93 100644 Binary files a/_module/ncs/cb_npcmeditat_hb.ncs and b/_module/ncs/cb_npcmeditat_hb.ncs differ diff --git a/_module/ncs/cb_npcsitchair.ncs b/_module/ncs/cb_npcsitchair.ncs index d201bbc..1325abb 100644 Binary files a/_module/ncs/cb_npcsitchair.ncs and b/_module/ncs/cb_npcsitchair.ncs differ diff --git a/_module/ncs/cb_npcsitchair2.ncs b/_module/ncs/cb_npcsitchair2.ncs index 1977062..fb393f6 100644 Binary files a/_module/ncs/cb_npcsitchair2.ncs and b/_module/ncs/cb_npcsitchair2.ncs differ diff --git a/_module/ncs/cb_npcsitchair3.ncs b/_module/ncs/cb_npcsitchair3.ncs index c53f438..c19d301 100644 Binary files a/_module/ncs/cb_npcsitchair3.ncs and b/_module/ncs/cb_npcsitchair3.ncs differ diff --git a/_module/ncs/cb_npcsitchair4.ncs b/_module/ncs/cb_npcsitchair4.ncs index f38b113..cfb41f8 100644 Binary files a/_module/ncs/cb_npcsitchair4.ncs and b/_module/ncs/cb_npcsitchair4.ncs differ diff --git a/_module/ncs/cb_npcworship_hb.ncs b/_module/ncs/cb_npcworship_hb.ncs index 4232363..c595377 100644 Binary files a/_module/ncs/cb_npcworship_hb.ncs and b/_module/ncs/cb_npcworship_hb.ncs differ diff --git a/_module/ncs/cb_onunequip.ncs b/_module/ncs/cb_onunequip.ncs index 63b1920..406736a 100644 Binary files a/_module/ncs/cb_onunequip.ncs and b/_module/ncs/cb_onunequip.ncs differ diff --git a/_module/ncs/cb_oz_chest.ncs b/_module/ncs/cb_oz_chest.ncs index c33f2d0..1618553 100644 Binary files a/_module/ncs/cb_oz_chest.ncs and b/_module/ncs/cb_oz_chest.ncs differ diff --git a/_module/ncs/cb_oz_heart.ncs b/_module/ncs/cb_oz_heart.ncs index 6d43cbb..8bc0c51 100644 Binary files a/_module/ncs/cb_oz_heart.ncs and b/_module/ncs/cb_oz_heart.ncs differ diff --git a/_module/ncs/cb_ozchk_1.ncs b/_module/ncs/cb_ozchk_1.ncs index 202fc02..1c9ce28 100644 Binary files a/_module/ncs/cb_ozchk_1.ncs and b/_module/ncs/cb_ozchk_1.ncs differ diff --git a/_module/ncs/cb_ozchk_2.ncs b/_module/ncs/cb_ozchk_2.ncs index c15c4ae..226cbb5 100644 Binary files a/_module/ncs/cb_ozchk_2.ncs and b/_module/ncs/cb_ozchk_2.ncs differ diff --git a/_module/ncs/cb_ozchk_3.ncs b/_module/ncs/cb_ozchk_3.ncs index 251a7c0..a8d14f5 100644 Binary files a/_module/ncs/cb_ozchk_3.ncs and b/_module/ncs/cb_ozchk_3.ncs differ diff --git a/_module/ncs/cb_ozchkrodsall.ncs b/_module/ncs/cb_ozchkrodsall.ncs index 497f6f2..3da4a86 100644 Binary files a/_module/ncs/cb_ozchkrodsall.ncs and b/_module/ncs/cb_ozchkrodsall.ncs differ diff --git a/_module/ncs/cb_ozentervfx.ncs b/_module/ncs/cb_ozentervfx.ncs index d2c14ed..83ad5eb 100644 Binary files a/_module/ncs/cb_ozentervfx.ncs and b/_module/ncs/cb_ozentervfx.ncs differ diff --git a/_module/ncs/cb_ozparty.ncs b/_module/ncs/cb_ozparty.ncs index 413bb69..3885495 100644 Binary files a/_module/ncs/cb_ozparty.ncs and b/_module/ncs/cb_ozparty.ncs differ diff --git a/_module/ncs/cb_ozport_01.ncs b/_module/ncs/cb_ozport_01.ncs index b625ecf..90c8a7d 100644 Binary files a/_module/ncs/cb_ozport_01.ncs and b/_module/ncs/cb_ozport_01.ncs differ diff --git a/_module/ncs/cb_ozport_33.ncs b/_module/ncs/cb_ozport_33.ncs index ef0394b..6bd30a3 100644 Binary files a/_module/ncs/cb_ozport_33.ncs and b/_module/ncs/cb_ozport_33.ncs differ diff --git a/_module/ncs/cb_ozrod01.ncs b/_module/ncs/cb_ozrod01.ncs index c43491c..5f2f59e 100644 Binary files a/_module/ncs/cb_ozrod01.ncs and b/_module/ncs/cb_ozrod01.ncs differ diff --git a/_module/ncs/cb_ozrod03.ncs b/_module/ncs/cb_ozrod03.ncs index 1dfb1e6..6a53486 100644 Binary files a/_module/ncs/cb_ozrod03.ncs and b/_module/ncs/cb_ozrod03.ncs differ diff --git a/_module/ncs/cb_ozrodblue.ncs b/_module/ncs/cb_ozrodblue.ncs index 46d47f4..57b5665 100644 Binary files a/_module/ncs/cb_ozrodblue.ncs and b/_module/ncs/cb_ozrodblue.ncs differ diff --git a/_module/ncs/cb_pally_01.ncs b/_module/ncs/cb_pally_01.ncs index 60b8502..a3b78dc 100644 Binary files a/_module/ncs/cb_pally_01.ncs and b/_module/ncs/cb_pally_01.ncs differ diff --git a/_module/ncs/cb_part_cain.ncs b/_module/ncs/cb_part_cain.ncs index fd1f4cf..6361cf6 100644 Binary files a/_module/ncs/cb_part_cain.ncs and b/_module/ncs/cb_part_cain.ncs differ diff --git a/_module/ncs/cb_passout.ncs b/_module/ncs/cb_passout.ncs index 00a5103..59c3e98 100644 Binary files a/_module/ncs/cb_passout.ncs and b/_module/ncs/cb_passout.ncs differ diff --git a/_module/ncs/cb_perc_orc.ncs b/_module/ncs/cb_perc_orc.ncs index f524f8a..d970c0f 100644 Binary files a/_module/ncs/cb_perc_orc.ncs and b/_module/ncs/cb_perc_orc.ncs differ diff --git a/_module/ncs/cb_perc_orc2.ncs b/_module/ncs/cb_perc_orc2.ncs index 7847c18..733d90d 100644 Binary files a/_module/ncs/cb_perc_orc2.ncs and b/_module/ncs/cb_perc_orc2.ncs differ diff --git a/_module/ncs/cb_perc_orc3.ncs b/_module/ncs/cb_perc_orc3.ncs index 687460f..70e99a0 100644 Binary files a/_module/ncs/cb_perc_orc3.ncs and b/_module/ncs/cb_perc_orc3.ncs differ diff --git a/_module/ncs/cb_perc_test.ncs b/_module/ncs/cb_perc_test.ncs index 9a643ad..9279216 100644 Binary files a/_module/ncs/cb_perc_test.ncs and b/_module/ncs/cb_perc_test.ncs differ diff --git a/_module/ncs/cb_port_trin.ncs b/_module/ncs/cb_port_trin.ncs index 2ed75e2..18a6871 100644 Binary files a/_module/ncs/cb_port_trin.ncs and b/_module/ncs/cb_port_trin.ncs differ diff --git a/_module/ncs/cb_portevil.ncs b/_module/ncs/cb_portevil.ncs index e0f6724..fb3d271 100644 Binary files a/_module/ncs/cb_portevil.ncs and b/_module/ncs/cb_portevil.ncs differ diff --git a/_module/ncs/cb_portevil2.ncs b/_module/ncs/cb_portevil2.ncs index e2cfa7a..8fc61f3 100644 Binary files a/_module/ncs/cb_portevil2.ncs and b/_module/ncs/cb_portevil2.ncs differ diff --git a/_module/ncs/cb_portgood.ncs b/_module/ncs/cb_portgood.ncs index 3819564..9fbdb59 100644 Binary files a/_module/ncs/cb_portgood.ncs and b/_module/ncs/cb_portgood.ncs differ diff --git a/_module/ncs/cb_portgood2.ncs b/_module/ncs/cb_portgood2.ncs index 035b243..2d0a6ca 100644 Binary files a/_module/ncs/cb_portgood2.ncs and b/_module/ncs/cb_portgood2.ncs differ diff --git a/_module/ncs/cb_pvp_chkitems.ncs b/_module/ncs/cb_pvp_chkitems.ncs index 23eb722..ce3facb 100644 Binary files a/_module/ncs/cb_pvp_chkitems.ncs and b/_module/ncs/cb_pvp_chkitems.ncs differ diff --git a/_module/ncs/cb_pvp_hb.ncs b/_module/ncs/cb_pvp_hb.ncs index dd787ff..cf5feef 100644 Binary files a/_module/ncs/cb_pvp_hb.ncs and b/_module/ncs/cb_pvp_hb.ncs differ diff --git a/_module/ncs/cb_pvp_portback.ncs b/_module/ncs/cb_pvp_portback.ncs index 563bf7d..dd9791b 100644 Binary files a/_module/ncs/cb_pvp_portback.ncs and b/_module/ncs/cb_pvp_portback.ncs differ diff --git a/_module/ncs/cb_pvpexit.ncs b/_module/ncs/cb_pvpexit.ncs index 45d79de..e09fbf0 100644 Binary files a/_module/ncs/cb_pvpexit.ncs and b/_module/ncs/cb_pvpexit.ncs differ diff --git a/_module/ncs/cb_pvpexit2.ncs b/_module/ncs/cb_pvpexit2.ncs index 570bd73..b293f95 100644 Binary files a/_module/ncs/cb_pvpexit2.ncs and b/_module/ncs/cb_pvpexit2.ncs differ diff --git a/_module/ncs/cb_pvptoken.ncs b/_module/ncs/cb_pvptoken.ncs index dd02f8b..0ffcffc 100644 Binary files a/_module/ncs/cb_pvptoken.ncs and b/_module/ncs/cb_pvptoken.ncs differ diff --git a/_module/ncs/cb_pvptoken2.ncs b/_module/ncs/cb_pvptoken2.ncs index bc8a682..a5cfda8 100644 Binary files a/_module/ncs/cb_pvptoken2.ncs and b/_module/ncs/cb_pvptoken2.ncs differ diff --git a/_module/ncs/cb_questchk_1.ncs b/_module/ncs/cb_questchk_1.ncs index ded2f3d..4894e68 100644 Binary files a/_module/ncs/cb_questchk_1.ncs and b/_module/ncs/cb_questchk_1.ncs differ diff --git a/_module/ncs/cb_ranger_r.ncs b/_module/ncs/cb_ranger_r.ncs index 082a55c..8bc6d26 100644 Binary files a/_module/ncs/cb_ranger_r.ncs and b/_module/ncs/cb_ranger_r.ncs differ diff --git a/_module/ncs/cb_ranger_w.ncs b/_module/ncs/cb_ranger_w.ncs index 77c86f8..bd65a3c 100644 Binary files a/_module/ncs/cb_ranger_w.ncs and b/_module/ncs/cb_ranger_w.ncs differ diff --git a/_module/ncs/cb_reb_monk15.ncs b/_module/ncs/cb_reb_monk15.ncs index 6a4f027..3682445 100644 Binary files a/_module/ncs/cb_reb_monk15.ncs and b/_module/ncs/cb_reb_monk15.ncs differ diff --git a/_module/ncs/cb_reb_party.ncs b/_module/ncs/cb_reb_party.ncs index 81d3098..3dadaf8 100644 Binary files a/_module/ncs/cb_reb_party.ncs and b/_module/ncs/cb_reb_party.ncs differ diff --git a/_module/ncs/cb_reb_partyport.ncs b/_module/ncs/cb_reb_partyport.ncs index 0e0a7c2..cc635f7 100644 Binary files a/_module/ncs/cb_reb_partyport.ncs and b/_module/ncs/cb_reb_partyport.ncs differ diff --git a/_module/ncs/cb_reb_percieve.ncs b/_module/ncs/cb_reb_percieve.ncs index a073938..431e502 100644 Binary files a/_module/ncs/cb_reb_percieve.ncs and b/_module/ncs/cb_reb_percieve.ncs differ diff --git a/_module/ncs/cb_reb_reborn.ncs b/_module/ncs/cb_reb_reborn.ncs index cba3e79..14f92ad 100644 Binary files a/_module/ncs/cb_reb_reborn.ncs and b/_module/ncs/cb_reb_reborn.ncs differ diff --git a/_module/ncs/cb_reb_trust.ncs b/_module/ncs/cb_reb_trust.ncs index 5d43f40..ec29e4e 100644 Binary files a/_module/ncs/cb_reb_trust.ncs and b/_module/ncs/cb_reb_trust.ncs differ diff --git a/_module/ncs/cb_rebornenter.ncs b/_module/ncs/cb_rebornenter.ncs index 50b02a5..e564e8b 100644 Binary files a/_module/ncs/cb_rebornenter.ncs and b/_module/ncs/cb_rebornenter.ncs differ diff --git a/_module/ncs/cb_rebspawn.ncs b/_module/ncs/cb_rebspawn.ncs index bdc009d..1ba9496 100644 Binary files a/_module/ncs/cb_rebspawn.ncs and b/_module/ncs/cb_rebspawn.ncs differ diff --git a/_module/ncs/cb_regate.ncs b/_module/ncs/cb_regate.ncs index 9111db2..2a1abfc 100644 Binary files a/_module/ncs/cb_regate.ncs and b/_module/ncs/cb_regate.ncs differ diff --git a/_module/ncs/cb_repentpally.ncs b/_module/ncs/cb_repentpally.ncs index b0667a5..14bb553 100644 Binary files a/_module/ncs/cb_repentpally.ncs and b/_module/ncs/cb_repentpally.ncs differ diff --git a/_module/ncs/cb_repentpally2.ncs b/_module/ncs/cb_repentpally2.ncs index 151a504..6f5a17c 100644 Binary files a/_module/ncs/cb_repentpally2.ncs and b/_module/ncs/cb_repentpally2.ncs differ diff --git a/_module/ncs/cb_reward_1000xp.ncs b/_module/ncs/cb_reward_1000xp.ncs index ac62d03..a10a600 100644 Binary files a/_module/ncs/cb_reward_1000xp.ncs and b/_module/ncs/cb_reward_1000xp.ncs differ diff --git a/_module/ncs/cb_reward_1500.ncs b/_module/ncs/cb_reward_1500.ncs index 11d14e0..9152c80 100644 Binary files a/_module/ncs/cb_reward_1500.ncs and b/_module/ncs/cb_reward_1500.ncs differ diff --git a/_module/ncs/cb_reward_500.ncs b/_module/ncs/cb_reward_500.ncs index 5c2c488..7eb28d8 100644 Binary files a/_module/ncs/cb_reward_500.ncs and b/_module/ncs/cb_reward_500.ncs differ diff --git a/_module/ncs/cb_reward_500xp.ncs b/_module/ncs/cb_reward_500xp.ncs index ac62d03..a10a600 100644 Binary files a/_module/ncs/cb_reward_500xp.ncs and b/_module/ncs/cb_reward_500xp.ncs differ diff --git a/_module/ncs/cb_rune1.ncs b/_module/ncs/cb_rune1.ncs index 04b5c98..cf42c9e 100644 Binary files a/_module/ncs/cb_rune1.ncs and b/_module/ncs/cb_rune1.ncs differ diff --git a/_module/ncs/cb_rune2.ncs b/_module/ncs/cb_rune2.ncs index 4e2b4ed..4689f64 100644 Binary files a/_module/ncs/cb_rune2.ncs and b/_module/ncs/cb_rune2.ncs differ diff --git a/_module/ncs/cb_setshogun.ncs b/_module/ncs/cb_setshogun.ncs index d1236da..e5522ca 100644 Binary files a/_module/ncs/cb_setshogun.ncs and b/_module/ncs/cb_setshogun.ncs differ diff --git a/_module/ncs/cb_shogunreward.ncs b/_module/ncs/cb_shogunreward.ncs index 39c91dc..732f632 100644 Binary files a/_module/ncs/cb_shogunreward.ncs and b/_module/ncs/cb_shogunreward.ncs differ diff --git a/_module/ncs/cb_shokey2.ncs b/_module/ncs/cb_shokey2.ncs index 1dcd50e..2141dcb 100644 Binary files a/_module/ncs/cb_shokey2.ncs and b/_module/ncs/cb_shokey2.ncs differ diff --git a/_module/ncs/cb_snowport.ncs b/_module/ncs/cb_snowport.ncs index 5deaf38..024962a 100644 Binary files a/_module/ncs/cb_snowport.ncs and b/_module/ncs/cb_snowport.ncs differ diff --git a/_module/ncs/cb_spawn_ambient.ncs b/_module/ncs/cb_spawn_ambient.ncs index 5cf6834..503cc35 100644 Binary files a/_module/ncs/cb_spawn_ambient.ncs and b/_module/ncs/cb_spawn_ambient.ncs differ diff --git a/_module/ncs/cb_spawn_search.ncs b/_module/ncs/cb_spawn_search.ncs index f289a12..c02138e 100644 Binary files a/_module/ncs/cb_spawn_search.ncs and b/_module/ncs/cb_spawn_search.ncs differ diff --git a/_module/ncs/cb_spawn_voice.ncs b/_module/ncs/cb_spawn_voice.ncs index 9afe3c7..c75fd7d 100644 Binary files a/_module/ncs/cb_spawn_voice.ncs and b/_module/ncs/cb_spawn_voice.ncs differ diff --git a/_module/ncs/cb_spawn_yield.ncs b/_module/ncs/cb_spawn_yield.ncs index 6cef987..9ba3bd5 100644 Binary files a/_module/ncs/cb_spawn_yield.ncs and b/_module/ncs/cb_spawn_yield.ncs differ diff --git a/_module/ncs/cb_spawncain.ncs b/_module/ncs/cb_spawncain.ncs index 44e1c5c..e4d2c73 100644 Binary files a/_module/ncs/cb_spawncain.ncs and b/_module/ncs/cb_spawncain.ncs differ diff --git a/_module/ncs/cb_spawnkidding.ncs b/_module/ncs/cb_spawnkidding.ncs index abe6462..fad7655 100644 Binary files a/_module/ncs/cb_spawnkidding.ncs and b/_module/ncs/cb_spawnkidding.ncs differ diff --git a/_module/ncs/cb_spawnmonks.ncs b/_module/ncs/cb_spawnmonks.ncs index 0780f36..fb54b40 100644 Binary files a/_module/ncs/cb_spawnmonks.ncs and b/_module/ncs/cb_spawnmonks.ncs differ diff --git a/_module/ncs/cb_start_bar.ncs b/_module/ncs/cb_start_bar.ncs index f301846..20132e5 100644 Binary files a/_module/ncs/cb_start_bar.ncs and b/_module/ncs/cb_start_bar.ncs differ diff --git a/_module/ncs/cb_startmax.ncs b/_module/ncs/cb_startmax.ncs index ae58161..eda0366 100644 Binary files a/_module/ncs/cb_startmax.ncs and b/_module/ncs/cb_startmax.ncs differ diff --git a/_module/ncs/cb_takebow.ncs b/_module/ncs/cb_takebow.ncs index 3d21816..89115f5 100644 Binary files a/_module/ncs/cb_takebow.ncs and b/_module/ncs/cb_takebow.ncs differ diff --git a/_module/ncs/cb_tke_25k.ncs b/_module/ncs/cb_tke_25k.ncs index 1049dbb..64bf263 100644 Binary files a/_module/ncs/cb_tke_25k.ncs and b/_module/ncs/cb_tke_25k.ncs differ diff --git a/_module/ncs/cb_tke_acidion.ncs b/_module/ncs/cb_tke_acidion.ncs index cfebecc..e1d69f7 100644 Binary files a/_module/ncs/cb_tke_acidion.ncs and b/_module/ncs/cb_tke_acidion.ncs differ diff --git a/_module/ncs/cb_tke_allplans.ncs b/_module/ncs/cb_tke_allplans.ncs index d3f7e53..113467e 100644 Binary files a/_module/ncs/cb_tke_allplans.ncs and b/_module/ncs/cb_tke_allplans.ncs differ diff --git a/_module/ncs/cb_tke_bookn.ncs b/_module/ncs/cb_tke_bookn.ncs index 11d1823..29e1aca 100644 Binary files a/_module/ncs/cb_tke_bookn.ncs and b/_module/ncs/cb_tke_bookn.ncs differ diff --git a/_module/ncs/cb_tke_dickie.ncs b/_module/ncs/cb_tke_dickie.ncs index 5bd0009..df47f44 100644 Binary files a/_module/ncs/cb_tke_dickie.ncs and b/_module/ncs/cb_tke_dickie.ncs differ diff --git a/_module/ncs/cb_tke_dragener.ncs b/_module/ncs/cb_tke_dragener.ncs index dfbddef..8091531 100644 Binary files a/_module/ncs/cb_tke_dragener.ncs and b/_module/ncs/cb_tke_dragener.ncs differ diff --git a/_module/ncs/cb_tke_drownoble.ncs b/_module/ncs/cb_tke_drownoble.ncs index b75a109..4a9e81e 100644 Binary files a/_module/ncs/cb_tke_drownoble.ncs and b/_module/ncs/cb_tke_drownoble.ncs differ diff --git a/_module/ncs/cb_tke_evilorb.ncs b/_module/ncs/cb_tke_evilorb.ncs index df7ac7f..5e1bcd2 100644 Binary files a/_module/ncs/cb_tke_evilorb.ncs and b/_module/ncs/cb_tke_evilorb.ncs differ diff --git a/_module/ncs/cb_tke_gemtool.ncs b/_module/ncs/cb_tke_gemtool.ncs index c737fa4..c64447f 100644 Binary files a/_module/ncs/cb_tke_gemtool.ncs and b/_module/ncs/cb_tke_gemtool.ncs differ diff --git a/_module/ncs/cb_tke_goodorb.ncs b/_module/ncs/cb_tke_goodorb.ncs index d5d00a6..b163100 100644 Binary files a/_module/ncs/cb_tke_goodorb.ncs and b/_module/ncs/cb_tke_goodorb.ncs differ diff --git a/_module/ncs/cb_tke_grail.ncs b/_module/ncs/cb_tke_grail.ncs index dbf1296..67fe8e0 100644 Binary files a/_module/ncs/cb_tke_grail.ncs and b/_module/ncs/cb_tke_grail.ncs differ diff --git a/_module/ncs/cb_tke_horn.ncs b/_module/ncs/cb_tke_horn.ncs index 04fbccc..f9e9fd0 100644 Binary files a/_module/ncs/cb_tke_horn.ncs and b/_module/ncs/cb_tke_horn.ncs differ diff --git a/_module/ncs/cb_tke_jimkey.ncs b/_module/ncs/cb_tke_jimkey.ncs index a68260c..5f29eb3 100644 Binary files a/_module/ncs/cb_tke_jimkey.ncs and b/_module/ncs/cb_tke_jimkey.ncs differ diff --git a/_module/ncs/cb_tke_lgrial.ncs b/_module/ncs/cb_tke_lgrial.ncs index e06d8d4..edc1f7c 100644 Binary files a/_module/ncs/cb_tke_lgrial.ncs and b/_module/ncs/cb_tke_lgrial.ncs differ diff --git a/_module/ncs/cb_tke_lstaff.ncs b/_module/ncs/cb_tke_lstaff.ncs index af7b84c..9ca36b0 100644 Binary files a/_module/ncs/cb_tke_lstaff.ncs and b/_module/ncs/cb_tke_lstaff.ncs differ diff --git a/_module/ncs/cb_tke_mahead.ncs b/_module/ncs/cb_tke_mahead.ncs index e3eee5e..69a0e07 100644 Binary files a/_module/ncs/cb_tke_mahead.ncs and b/_module/ncs/cb_tke_mahead.ncs differ diff --git a/_module/ncs/cb_tke_mug.ncs b/_module/ncs/cb_tke_mug.ncs index 1133dc2..58b26d8 100644 Binary files a/_module/ncs/cb_tke_mug.ncs and b/_module/ncs/cb_tke_mug.ncs differ diff --git a/_module/ncs/cb_tke_ore.ncs b/_module/ncs/cb_tke_ore.ncs index 76739e3..7063698 100644 Binary files a/_module/ncs/cb_tke_ore.ncs and b/_module/ncs/cb_tke_ore.ncs differ diff --git a/_module/ncs/cb_tke_ozrods.ncs b/_module/ncs/cb_tke_ozrods.ncs index 3ee1193..8dbb8d0 100644 Binary files a/_module/ncs/cb_tke_ozrods.ncs and b/_module/ncs/cb_tke_ozrods.ncs differ diff --git a/_module/ncs/cb_tke_serum.ncs b/_module/ncs/cb_tke_serum.ncs index 6a05dfa..3f935b9 100644 Binary files a/_module/ncs/cb_tke_serum.ncs and b/_module/ncs/cb_tke_serum.ncs differ diff --git a/_module/ncs/cb_tke_taghead.ncs b/_module/ncs/cb_tke_taghead.ncs index f7c3023..63b96f2 100644 Binary files a/_module/ncs/cb_tke_taghead.ncs and b/_module/ncs/cb_tke_taghead.ncs differ diff --git a/_module/ncs/cb_tke_tmbattle.ncs b/_module/ncs/cb_tke_tmbattle.ncs index 7da6a0d..def6f53 100644 Binary files a/_module/ncs/cb_tke_tmbattle.ncs and b/_module/ncs/cb_tke_tmbattle.ncs differ diff --git a/_module/ncs/cb_tke_yoga.ncs b/_module/ncs/cb_tke_yoga.ncs index 203b0ee..798130c 100644 Binary files a/_module/ncs/cb_tke_yoga.ncs and b/_module/ncs/cb_tke_yoga.ncs differ diff --git a/_module/ncs/cb_traner1.ncs b/_module/ncs/cb_traner1.ncs index 438ed60..beff1a2 100644 Binary files a/_module/ncs/cb_traner1.ncs and b/_module/ncs/cb_traner1.ncs differ diff --git a/_module/ncs/cb_trapstore.ncs b/_module/ncs/cb_trapstore.ncs index 5f0ce14..9002053 100644 Binary files a/_module/ncs/cb_trapstore.ncs and b/_module/ncs/cb_trapstore.ncs differ diff --git a/_module/ncs/cb_updates.ncs b/_module/ncs/cb_updates.ncs index c5a2ec9..dff560d 100644 Binary files a/_module/ncs/cb_updates.ncs and b/_module/ncs/cb_updates.ncs differ diff --git a/_module/ncs/cb_vis_cain.ncs b/_module/ncs/cb_vis_cain.ncs index 77d5b04..3065e2e 100644 Binary files a/_module/ncs/cb_vis_cain.ncs and b/_module/ncs/cb_vis_cain.ncs differ diff --git a/_module/ncs/cb_vis_cloak.ncs b/_module/ncs/cb_vis_cloak.ncs index 2c650c2..703bda4 100644 Binary files a/_module/ncs/cb_vis_cloak.ncs and b/_module/ncs/cb_vis_cloak.ncs differ diff --git a/_module/ncs/cb_vis_lith.ncs b/_module/ncs/cb_vis_lith.ncs index 87c5405..d6d0916 100644 Binary files a/_module/ncs/cb_vis_lith.ncs and b/_module/ncs/cb_vis_lith.ncs differ diff --git a/_module/ncs/cb_vis_monk.ncs b/_module/ncs/cb_vis_monk.ncs index 8bc63cb..c8d7bbe 100644 Binary files a/_module/ncs/cb_vis_monk.ncs and b/_module/ncs/cb_vis_monk.ncs differ diff --git a/_module/ncs/cb_vis_pally.ncs b/_module/ncs/cb_vis_pally.ncs index 1e28800..ba35457 100644 Binary files a/_module/ncs/cb_vis_pally.ncs and b/_module/ncs/cb_vis_pally.ncs differ diff --git a/_module/ncs/cb_vis_reb2.ncs b/_module/ncs/cb_vis_reb2.ncs index 1e3604d..e6bbd3f 100644 Binary files a/_module/ncs/cb_vis_reb2.ncs and b/_module/ncs/cb_vis_reb2.ncs differ diff --git a/_module/ncs/cb_vis_test.ncs b/_module/ncs/cb_vis_test.ncs index 06125ea..b914e1e 100644 Binary files a/_module/ncs/cb_vis_test.ncs and b/_module/ncs/cb_vis_test.ncs differ diff --git a/_module/ncs/cb_vis_used.ncs b/_module/ncs/cb_vis_used.ncs index e08c395..3105c2e 100644 Binary files a/_module/ncs/cb_vis_used.ncs and b/_module/ncs/cb_vis_used.ncs differ diff --git a/_module/ncs/cb_welcome.ncs b/_module/ncs/cb_welcome.ncs index c49ecdc..ef489da 100644 Binary files a/_module/ncs/cb_welcome.ncs and b/_module/ncs/cb_welcome.ncs differ diff --git a/_module/ncs/cb_wellport.ncs b/_module/ncs/cb_wellport.ncs index 2448a12..13594cf 100644 Binary files a/_module/ncs/cb_wellport.ncs and b/_module/ncs/cb_wellport.ncs differ diff --git a/_module/ncs/cb_x2_per_reb.ncs b/_module/ncs/cb_x2_per_reb.ncs index e0a13cc..2bf3b21 100644 Binary files a/_module/ncs/cb_x2_per_reb.ncs and b/_module/ncs/cb_x2_per_reb.ncs differ diff --git a/_module/ncs/cb_x2_percasy.ncs b/_module/ncs/cb_x2_percasy.ncs index 7540972..3cc5248 100644 Binary files a/_module/ncs/cb_x2_percasy.ncs and b/_module/ncs/cb_x2_percasy.ncs differ diff --git a/_module/ncs/cb_x2_percieve.ncs b/_module/ncs/cb_x2_percieve.ncs index 9b0f7f7..51f250d 100644 Binary files a/_module/ncs/cb_x2_percieve.ncs and b/_module/ncs/cb_x2_percieve.ncs differ diff --git a/_module/ncs/cb_x2_percieve2.ncs b/_module/ncs/cb_x2_percieve2.ncs index 7310020..7e9b304 100644 Binary files a/_module/ncs/cb_x2_percieve2.ncs and b/_module/ncs/cb_x2_percieve2.ncs differ diff --git a/_module/ncs/cb_x2_percieve3.ncs b/_module/ncs/cb_x2_percieve3.ncs index 30cc7a0..09e918e 100644 Binary files a/_module/ncs/cb_x2_percieve3.ncs and b/_module/ncs/cb_x2_percieve3.ncs differ diff --git a/_module/ncs/cb_x2_percieve4.ncs b/_module/ncs/cb_x2_percieve4.ncs index 0c11b6f..cda01fd 100644 Binary files a/_module/ncs/cb_x2_percieve4.ncs and b/_module/ncs/cb_x2_percieve4.ncs differ diff --git a/_module/ncs/cb_x2_search.ncs b/_module/ncs/cb_x2_search.ncs index f289a12..c02138e 100644 Binary files a/_module/ncs/cb_x2_search.ncs and b/_module/ncs/cb_x2_search.ncs differ diff --git a/_module/ncs/cb_yield.ncs b/_module/ncs/cb_yield.ncs index 80b44a8..40e194b 100644 Binary files a/_module/ncs/cb_yield.ncs and b/_module/ncs/cb_yield.ncs differ diff --git a/_module/ncs/cb_yield_attack.ncs b/_module/ncs/cb_yield_attack.ncs index deeb2cd..6a50f59 100644 Binary files a/_module/ncs/cb_yield_attack.ncs and b/_module/ncs/cb_yield_attack.ncs differ diff --git a/_module/ncs/chair_sit.ncs b/_module/ncs/chair_sit.ncs index 607923d..97b7996 100644 Binary files a/_module/ncs/chair_sit.ncs and b/_module/ncs/chair_sit.ncs differ diff --git a/_module/ncs/clean_store.ncs b/_module/ncs/clean_store.ncs index ea12faf..69f2f28 100644 Binary files a/_module/ncs/clean_store.ncs and b/_module/ncs/clean_store.ncs differ diff --git a/_module/ncs/cloakpotion.ncs b/_module/ncs/cloakpotion.ncs index c0d0af9..9d83135 100644 Binary files a/_module/ncs/cloakpotion.ncs and b/_module/ncs/cloakpotion.ncs differ diff --git a/_module/ncs/close_field1.ncs b/_module/ncs/close_field1.ncs index b2cba0a..350407a 100644 Binary files a/_module/ncs/close_field1.ncs and b/_module/ncs/close_field1.ncs differ diff --git a/_module/ncs/close_field2.ncs b/_module/ncs/close_field2.ncs index 32fca12..054857d 100644 Binary files a/_module/ncs/close_field2.ncs and b/_module/ncs/close_field2.ncs differ diff --git a/_module/ncs/close_field3.ncs b/_module/ncs/close_field3.ncs index c267ede..470fa3a 100644 Binary files a/_module/ncs/close_field3.ncs and b/_module/ncs/close_field3.ncs differ diff --git a/_module/ncs/close_field4.ncs b/_module/ncs/close_field4.ncs index 93ebc81..d6c0189 100644 Binary files a/_module/ncs/close_field4.ncs and b/_module/ncs/close_field4.ncs differ diff --git a/_module/ncs/close_field5.ncs b/_module/ncs/close_field5.ncs index 98c27b6..9b9d762 100644 Binary files a/_module/ncs/close_field5.ncs and b/_module/ncs/close_field5.ncs differ diff --git a/_module/ncs/close_field6.ncs b/_module/ncs/close_field6.ncs index a16e8ec..4bd695f 100644 Binary files a/_module/ncs/close_field6.ncs and b/_module/ncs/close_field6.ncs differ diff --git a/_module/ncs/closedoors.ncs b/_module/ncs/closedoors.ncs index c3fd294..6affbcc 100644 Binary files a/_module/ncs/closedoors.ncs and b/_module/ncs/closedoors.ncs differ diff --git a/_module/ncs/cn_npc_read.ncs b/_module/ncs/cn_npc_read.ncs index e29f428..1faf024 100644 Binary files a/_module/ncs/cn_npc_read.ncs and b/_module/ncs/cn_npc_read.ncs differ diff --git a/_module/ncs/cold_trap_enter.ncs b/_module/ncs/cold_trap_enter.ncs index 991f952..1b1106c 100644 Binary files a/_module/ncs/cold_trap_enter.ncs and b/_module/ncs/cold_trap_enter.ncs differ diff --git a/_module/ncs/conv_atlantis_p2.ncs b/_module/ncs/conv_atlantis_p2.ncs index 88b2180..7dcf013 100644 Binary files a/_module/ncs/conv_atlantis_p2.ncs and b/_module/ncs/conv_atlantis_p2.ncs differ diff --git a/_module/ncs/conv_bvgrd_rndm1.ncs b/_module/ncs/conv_bvgrd_rndm1.ncs index ba7cfd8..4de5567 100644 Binary files a/_module/ncs/conv_bvgrd_rndm1.ncs and b/_module/ncs/conv_bvgrd_rndm1.ncs differ diff --git a/_module/ncs/conv_bvgrd_rndm2.ncs b/_module/ncs/conv_bvgrd_rndm2.ncs index ba7cfd8..4de5567 100644 Binary files a/_module/ncs/conv_bvgrd_rndm2.ncs and b/_module/ncs/conv_bvgrd_rndm2.ncs differ diff --git a/_module/ncs/council_on_enter.ncs b/_module/ncs/council_on_enter.ncs index 0344286..91c0410 100644 Binary files a/_module/ncs/council_on_enter.ncs and b/_module/ncs/council_on_enter.ncs differ diff --git a/_module/ncs/council_port.ncs b/_module/ncs/council_port.ncs index 980a9f5..13f3e1c 100644 Binary files a/_module/ncs/council_port.ncs and b/_module/ncs/council_port.ncs differ diff --git a/_module/ncs/council_rndm1.ncs b/_module/ncs/council_rndm1.ncs index ba7cfd8..4de5567 100644 Binary files a/_module/ncs/council_rndm1.ncs and b/_module/ncs/council_rndm1.ncs differ diff --git a/_module/ncs/cs_canbuysand.ncs b/_module/ncs/cs_canbuysand.ncs index c67c269..133ea68 100644 Binary files a/_module/ncs/cs_canbuysand.ncs and b/_module/ncs/cs_canbuysand.ncs differ diff --git a/_module/ncs/cs_d_hasstolen.ncs b/_module/ncs/cs_d_hasstolen.ncs index 5d475d1..f9d3954 100644 Binary files a/_module/ncs/cs_d_hasstolen.ncs and b/_module/ncs/cs_d_hasstolen.ncs differ diff --git a/_module/ncs/cs_debugoff.ncs b/_module/ncs/cs_debugoff.ncs index 44061a8..0def326 100644 Binary files a/_module/ncs/cs_debugoff.ncs and b/_module/ncs/cs_debugoff.ncs differ diff --git a/_module/ncs/cs_debugon.ncs b/_module/ncs/cs_debugon.ncs index e82ae03..404b235 100644 Binary files a/_module/ncs/cs_debugon.ncs and b/_module/ncs/cs_debugon.ncs differ diff --git a/_module/ncs/cs_dmareatrigger.ncs b/_module/ncs/cs_dmareatrigger.ncs index e0793ff..aa8cc9c 100644 Binary files a/_module/ncs/cs_dmareatrigger.ncs and b/_module/ncs/cs_dmareatrigger.ncs differ diff --git a/_module/ncs/cs_invul_toggle.ncs b/_module/ncs/cs_invul_toggle.ncs index d8e83f6..8084f22 100644 Binary files a/_module/ncs/cs_invul_toggle.ncs and b/_module/ncs/cs_invul_toggle.ncs differ diff --git a/_module/ncs/cs_loot_high.ncs b/_module/ncs/cs_loot_high.ncs index 093ded1..858741a 100644 Binary files a/_module/ncs/cs_loot_high.ncs and b/_module/ncs/cs_loot_high.ncs differ diff --git a/_module/ncs/cs_loot_low.ncs b/_module/ncs/cs_loot_low.ncs index c2d260e..acf676a 100644 Binary files a/_module/ncs/cs_loot_low.ncs and b/_module/ncs/cs_loot_low.ncs differ diff --git a/_module/ncs/cs_loot_med.ncs b/_module/ncs/cs_loot_med.ncs index 0ca4722..4a59fc8 100644 Binary files a/_module/ncs/cs_loot_med.ncs and b/_module/ncs/cs_loot_med.ncs differ diff --git a/_module/ncs/cs_loot_monster.ncs b/_module/ncs/cs_loot_monster.ncs index 3dea4a2..a688195 100644 Binary files a/_module/ncs/cs_loot_monster.ncs and b/_module/ncs/cs_loot_monster.ncs differ diff --git a/_module/ncs/cs_loot_unique.ncs b/_module/ncs/cs_loot_unique.ncs index a9f799e..d1633af 100644 Binary files a/_module/ncs/cs_loot_unique.ncs and b/_module/ncs/cs_loot_unique.ncs differ diff --git a/_module/ncs/cs_port2appraise.ncs b/_module/ncs/cs_port2appraise.ncs index 6879263..70da313 100644 Binary files a/_module/ncs/cs_port2appraise.ncs and b/_module/ncs/cs_port2appraise.ncs differ diff --git a/_module/ncs/cs_port2lwrcaves.ncs b/_module/ncs/cs_port2lwrcaves.ncs index c959357..a70a92e 100644 Binary files a/_module/ncs/cs_port2lwrcaves.ncs and b/_module/ncs/cs_port2lwrcaves.ncs differ diff --git a/_module/ncs/cs_port2trinity.ncs b/_module/ncs/cs_port2trinity.ncs index a2111c9..8c932a4 100644 Binary files a/_module/ncs/cs_port2trinity.ncs and b/_module/ncs/cs_port2trinity.ncs differ diff --git a/_module/ncs/cs_resetpcitems.ncs b/_module/ncs/cs_resetpcitems.ncs index 40a82c7..80f4a8f 100644 Binary files a/_module/ncs/cs_resetpcitems.ncs and b/_module/ncs/cs_resetpcitems.ncs differ diff --git a/_module/ncs/cs_symbolofmight.ncs b/_module/ncs/cs_symbolofmight.ncs index d5bbb8b..4fe3661 100644 Binary files a/_module/ncs/cs_symbolofmight.ncs and b/_module/ncs/cs_symbolofmight.ncs differ diff --git a/_module/ncs/cs_takegold.ncs b/_module/ncs/cs_takegold.ncs index fe71a79..1969790 100644 Binary files a/_module/ncs/cs_takegold.ncs and b/_module/ncs/cs_takegold.ncs differ diff --git a/_module/ncs/cv_inn_door.ncs b/_module/ncs/cv_inn_door.ncs new file mode 100644 index 0000000..06b157b Binary files /dev/null and b/_module/ncs/cv_inn_door.ncs differ diff --git a/_module/ncs/damage_statue.ncs b/_module/ncs/damage_statue.ncs index 3ca9a8c..1e1dba0 100644 Binary files a/_module/ncs/damage_statue.ncs and b/_module/ncs/damage_statue.ncs differ diff --git a/_module/ncs/dexring_gen.ncs b/_module/ncs/dexring_gen.ncs index 6ca884b..c1fffd6 100644 Binary files a/_module/ncs/dexring_gen.ncs and b/_module/ncs/dexring_gen.ncs differ diff --git a/_module/ncs/dm_throne.ncs b/_module/ncs/dm_throne.ncs index 91e03c7..49bce04 100644 Binary files a/_module/ncs/dm_throne.ncs and b/_module/ncs/dm_throne.ncs differ diff --git a/_module/ncs/dmc_containergen.ncs b/_module/ncs/dmc_containergen.ncs index b8440bc..b30b067 100644 Binary files a/_module/ncs/dmc_containergen.ncs and b/_module/ncs/dmc_containergen.ncs differ diff --git a/_module/ncs/dmshelper.ncs b/_module/ncs/dmshelper.ncs index c3806d6..190d481 100644 Binary files a/_module/ncs/dmshelper.ncs and b/_module/ncs/dmshelper.ncs differ diff --git a/_module/ncs/doorforceunlocdc.ncs b/_module/ncs/doorforceunlocdc.ncs index 27041f5..4909938 100644 Binary files a/_module/ncs/doorforceunlocdc.ncs and b/_module/ncs/doorforceunlocdc.ncs differ diff --git a/_module/ncs/dreamcatcher.ncs b/_module/ncs/dreamcatcher.ncs index adc3ad2..66d4553 100644 Binary files a/_module/ncs/dreamcatcher.ncs and b/_module/ncs/dreamcatcher.ncs differ diff --git a/_module/ncs/drink_n_explode3.ncs b/_module/ncs/drink_n_explode3.ncs index 21433b3..e839a09 100644 Binary files a/_module/ncs/drink_n_explode3.ncs and b/_module/ncs/drink_n_explode3.ncs differ diff --git a/_module/ncs/elec_trap_enter.ncs b/_module/ncs/elec_trap_enter.ncs index e0eff76..ec2a0c9 100644 Binary files a/_module/ncs/elec_trap_enter.ncs and b/_module/ncs/elec_trap_enter.ncs differ diff --git a/_module/ncs/elrie_stat_trig.ncs b/_module/ncs/elrie_stat_trig.ncs index ea49df9..deb67d4 100644 Binary files a/_module/ncs/elrie_stat_trig.ncs and b/_module/ncs/elrie_stat_trig.ncs differ diff --git a/_module/ncs/farmer_give_gold.ncs b/_module/ncs/farmer_give_gold.ncs index dd1ec70..0058b66 100644 Binary files a/_module/ncs/farmer_give_gold.ncs and b/_module/ncs/farmer_give_gold.ncs differ diff --git a/_module/ncs/farmer_item_chk.ncs b/_module/ncs/farmer_item_chk.ncs index a0af9bd..bd365a5 100644 Binary files a/_module/ncs/farmer_item_chk.ncs and b/_module/ncs/farmer_item_chk.ncs differ diff --git a/_module/ncs/farmer_take_item.ncs b/_module/ncs/farmer_take_item.ncs index 63f5fe5..1ec5c91 100644 Binary files a/_module/ncs/farmer_take_item.ncs and b/_module/ncs/farmer_take_item.ncs differ diff --git a/_module/ncs/farmer_var_chk.ncs b/_module/ncs/farmer_var_chk.ncs index 312605d..0f47bd5 100644 Binary files a/_module/ncs/farmer_var_chk.ncs and b/_module/ncs/farmer_var_chk.ncs differ diff --git a/_module/ncs/ffx_fc_exit.ncs b/_module/ncs/ffx_fc_exit.ncs index ac1bbdb..f5d2a6c 100644 Binary files a/_module/ncs/ffx_fc_exit.ncs and b/_module/ncs/ffx_fc_exit.ncs differ diff --git a/_module/ncs/ffx_fc_lever.ncs b/_module/ncs/ffx_fc_lever.ncs index 5e6fbb6..b9d7778 100644 Binary files a/_module/ncs/ffx_fc_lever.ncs and b/_module/ncs/ffx_fc_lever.ncs differ diff --git a/_module/ncs/ffx_fc_lever2.ncs b/_module/ncs/ffx_fc_lever2.ncs index 3d0175b..b5cd853 100644 Binary files a/_module/ncs/ffx_fc_lever2.ncs and b/_module/ncs/ffx_fc_lever2.ncs differ diff --git a/_module/ncs/ffx_fc_lever3.ncs b/_module/ncs/ffx_fc_lever3.ncs index 99fd17b..0509210 100644 Binary files a/_module/ncs/ffx_fc_lever3.ncs and b/_module/ncs/ffx_fc_lever3.ncs differ diff --git a/_module/ncs/ffx_fc_lever9.ncs b/_module/ncs/ffx_fc_lever9.ncs index ad875c7..1497c39 100644 Binary files a/_module/ncs/ffx_fc_lever9.ncs and b/_module/ncs/ffx_fc_lever9.ncs differ diff --git a/_module/ncs/ffx_fc_outside.ncs b/_module/ncs/ffx_fc_outside.ncs index 4a024c7..aff21e9 100644 Binary files a/_module/ncs/ffx_fc_outside.ncs and b/_module/ncs/ffx_fc_outside.ncs differ diff --git a/_module/ncs/fire_trap_enter.ncs b/_module/ncs/fire_trap_enter.ncs index 1875e57..bc57af7 100644 Binary files a/_module/ncs/fire_trap_enter.ncs and b/_module/ncs/fire_trap_enter.ncs differ diff --git a/_module/ncs/frontgatelocktxt.ncs b/_module/ncs/frontgatelocktxt.ncs index e594fb9..da8d6dd 100644 Binary files a/_module/ncs/frontgatelocktxt.ncs and b/_module/ncs/frontgatelocktxt.ncs differ diff --git a/_module/ncs/frothic_portal.ncs b/_module/ncs/frothic_portal.ncs index f924102..952f63f 100644 Binary files a/_module/ncs/frothic_portal.ncs and b/_module/ncs/frothic_portal.ncs differ diff --git a/_module/ncs/frthrjez_var_chk.ncs b/_module/ncs/frthrjez_var_chk.ncs index 0bb5b32..6fdb3d6 100644 Binary files a/_module/ncs/frthrjez_var_chk.ncs and b/_module/ncs/frthrjez_var_chk.ncs differ diff --git a/_module/ncs/gen_loot_bodak.ncs b/_module/ncs/gen_loot_bodak.ncs index 4685bcd..b65aba3 100644 Binary files a/_module/ncs/gen_loot_bodak.ncs and b/_module/ncs/gen_loot_bodak.ncs differ diff --git a/_module/ncs/gen_loot_ck.ncs b/_module/ncs/gen_loot_ck.ncs index 7b92e08..51d2ce4 100644 Binary files a/_module/ncs/gen_loot_ck.ncs and b/_module/ncs/gen_loot_ck.ncs differ diff --git a/_module/ncs/gen_loot_eleven.ncs b/_module/ncs/gen_loot_eleven.ncs index 29a6e40..04f810e 100644 Binary files a/_module/ncs/gen_loot_eleven.ncs and b/_module/ncs/gen_loot_eleven.ncs differ diff --git a/_module/ncs/gen_loot_jagang.ncs b/_module/ncs/gen_loot_jagang.ncs index 1a4598f..2546b26 100644 Binary files a/_module/ncs/gen_loot_jagang.ncs and b/_module/ncs/gen_loot_jagang.ncs differ diff --git a/_module/ncs/gen_loot_ooze.ncs b/_module/ncs/gen_loot_ooze.ncs index 29a6e40..04f810e 100644 Binary files a/_module/ncs/gen_loot_ooze.ncs and b/_module/ncs/gen_loot_ooze.ncs differ diff --git a/_module/ncs/gen_loot_pawn.ncs b/_module/ncs/gen_loot_pawn.ncs index 29a6e40..04f810e 100644 Binary files a/_module/ncs/gen_loot_pawn.ncs and b/_module/ncs/gen_loot_pawn.ncs differ diff --git a/_module/ncs/gen_loot_sentros.ncs b/_module/ncs/gen_loot_sentros.ncs index 7b92e08..51d2ce4 100644 Binary files a/_module/ncs/gen_loot_sentros.ncs and b/_module/ncs/gen_loot_sentros.ncs differ diff --git a/_module/ncs/gen_loot_vassi.ncs b/_module/ncs/gen_loot_vassi.ncs index 7b92e08..51d2ce4 100644 Binary files a/_module/ncs/gen_loot_vassi.ncs and b/_module/ncs/gen_loot_vassi.ncs differ diff --git a/_module/ncs/globeofpower.ncs b/_module/ncs/globeofpower.ncs index 425db5e..4402bd4 100644 Binary files a/_module/ncs/globeofpower.ncs and b/_module/ncs/globeofpower.ncs differ diff --git a/_module/ncs/glor_stat_trig.ncs b/_module/ncs/glor_stat_trig.ncs index 0702cf9..b88e38e 100644 Binary files a/_module/ncs/glor_stat_trig.ncs and b/_module/ncs/glor_stat_trig.ncs differ diff --git a/_module/ncs/gratch_zedd_port.ncs b/_module/ncs/gratch_zedd_port.ncs index 3a6bea6..329e7b8 100644 Binary files a/_module/ncs/gratch_zedd_port.ncs and b/_module/ncs/gratch_zedd_port.ncs differ diff --git a/_module/ncs/grend_item_chk.ncs b/_module/ncs/grend_item_chk.ncs index 556ded9..35e860b 100644 Binary files a/_module/ncs/grend_item_chk.ncs and b/_module/ncs/grend_item_chk.ncs differ diff --git a/_module/ncs/grend_item_chk2.ncs b/_module/ncs/grend_item_chk2.ncs index 1c84967..fe38df9 100644 Binary files a/_module/ncs/grend_item_chk2.ncs and b/_module/ncs/grend_item_chk2.ncs differ diff --git a/_module/ncs/grend_item_chk3.ncs b/_module/ncs/grend_item_chk3.ncs index e0b50f1..dd187a1 100644 Binary files a/_module/ncs/grend_item_chk3.ncs and b/_module/ncs/grend_item_chk3.ncs differ diff --git a/_module/ncs/grend_item_take.ncs b/_module/ncs/grend_item_take.ncs index 14a01ed..cb7cfdc 100644 Binary files a/_module/ncs/grend_item_take.ncs and b/_module/ncs/grend_item_take.ncs differ diff --git a/_module/ncs/grend_var_chk.ncs b/_module/ncs/grend_var_chk.ncs index ed41b19..0d8af48 100644 Binary files a/_module/ncs/grend_var_chk.ncs and b/_module/ncs/grend_var_chk.ncs differ diff --git a/_module/ncs/grndkpr_var_chk.ncs b/_module/ncs/grndkpr_var_chk.ncs index c98788c..11fe6b7 100644 Binary files a/_module/ncs/grndkpr_var_chk.ncs and b/_module/ncs/grndkpr_var_chk.ncs differ diff --git a/_module/ncs/hidden_trapdoor.ncs b/_module/ncs/hidden_trapdoor.ncs index 0820299..7369581 100644 Binary files a/_module/ncs/hidden_trapdoor.ncs and b/_module/ncs/hidden_trapdoor.ncs differ diff --git a/_module/ncs/hif_onmoduleload.ncs b/_module/ncs/hif_onmoduleload.ncs index 9d0d300..cc8a43d 100644 Binary files a/_module/ncs/hif_onmoduleload.ncs and b/_module/ncs/hif_onmoduleload.ncs differ diff --git a/_module/ncs/holywater_on_use.ncs b/_module/ncs/holywater_on_use.ncs index 0a2f8eb..7b309ae 100644 Binary files a/_module/ncs/holywater_on_use.ncs and b/_module/ncs/holywater_on_use.ncs differ diff --git a/_module/ncs/jagang_recall.ncs b/_module/ncs/jagang_recall.ncs index 24f6f23..b519fab 100644 Binary files a/_module/ncs/jagang_recall.ncs and b/_module/ncs/jagang_recall.ncs differ diff --git a/_module/ncs/jano_activate.ncs b/_module/ncs/jano_activate.ncs index 437e645..8402d29 100644 Binary files a/_module/ncs/jano_activate.ncs and b/_module/ncs/jano_activate.ncs differ diff --git a/_module/ncs/jano_death.ncs b/_module/ncs/jano_death.ncs index d097b0d..52ba820 100644 Binary files a/_module/ncs/jano_death.ncs and b/_module/ncs/jano_death.ncs differ diff --git a/_module/ncs/jano_heartbeat.ncs b/_module/ncs/jano_heartbeat.ncs index 090ccaf..10f388a 100644 Binary files a/_module/ncs/jano_heartbeat.ncs and b/_module/ncs/jano_heartbeat.ncs differ diff --git a/_module/ncs/jano_ondying.ncs b/_module/ncs/jano_ondying.ncs index 9345e33..38ef204 100644 Binary files a/_module/ncs/jano_ondying.ncs and b/_module/ncs/jano_ondying.ncs differ diff --git a/_module/ncs/jano_onenter.ncs b/_module/ncs/jano_onenter.ncs index 7753d91..245d2e4 100644 Binary files a/_module/ncs/jano_onenter.ncs and b/_module/ncs/jano_onenter.ncs differ diff --git a/_module/ncs/jano_onrest.ncs b/_module/ncs/jano_onrest.ncs index 8393d48..73c467f 100644 Binary files a/_module/ncs/jano_onrest.ncs and b/_module/ncs/jano_onrest.ncs differ diff --git a/_module/ncs/jano_respawn.ncs b/_module/ncs/jano_respawn.ncs index 4358d7f..a10540a 100644 Binary files a/_module/ncs/jano_respawn.ncs and b/_module/ncs/jano_respawn.ncs differ diff --git a/_module/ncs/keyofcheating.ncs b/_module/ncs/keyofcheating.ncs index 0f7edd9..e073ed6 100644 Binary files a/_module/ncs/keyofcheating.ncs and b/_module/ncs/keyofcheating.ncs differ diff --git a/_module/ncs/librar_item_chk.ncs b/_module/ncs/librar_item_chk.ncs index f1d27c1..a7ee7b0 100644 Binary files a/_module/ncs/librar_item_chk.ncs and b/_module/ncs/librar_item_chk.ncs differ diff --git a/_module/ncs/librar_item_chk2.ncs b/_module/ncs/librar_item_chk2.ncs index 0bf1daf..db880ea 100644 Binary files a/_module/ncs/librar_item_chk2.ncs and b/_module/ncs/librar_item_chk2.ncs differ diff --git a/_module/ncs/librar_item_tak2.ncs b/_module/ncs/librar_item_tak2.ncs index c63a558..b1f44b7 100644 Binary files a/_module/ncs/librar_item_tak2.ncs and b/_module/ncs/librar_item_tak2.ncs differ diff --git a/_module/ncs/librar_take_card.ncs b/_module/ncs/librar_take_card.ncs index 793226a..9c3f958 100644 Binary files a/_module/ncs/librar_take_card.ncs and b/_module/ncs/librar_take_card.ncs differ diff --git a/_module/ncs/librar_var_chk2.ncs b/_module/ncs/librar_var_chk2.ncs index 3aa2368..cb733e5 100644 Binary files a/_module/ncs/librar_var_chk2.ncs and b/_module/ncs/librar_var_chk2.ncs differ diff --git a/_module/ncs/librar_var_set2.ncs b/_module/ncs/librar_var_set2.ncs index 4179c28..45d2f99 100644 Binary files a/_module/ncs/librar_var_set2.ncs and b/_module/ncs/librar_var_set2.ncs differ diff --git a/_module/ncs/library_item_tak.ncs b/_module/ncs/library_item_tak.ncs index fd65303..702efcc 100644 Binary files a/_module/ncs/library_item_tak.ncs and b/_module/ncs/library_item_tak.ncs differ diff --git a/_module/ncs/library_var_chk.ncs b/_module/ncs/library_var_chk.ncs index ee6ffc9..085a189 100644 Binary files a/_module/ncs/library_var_chk.ncs and b/_module/ncs/library_var_chk.ncs differ diff --git a/_module/ncs/loot_notify.ncs b/_module/ncs/loot_notify.ncs index fe1a24f..f185b47 100644 Binary files a/_module/ncs/loot_notify.ncs and b/_module/ncs/loot_notify.ncs differ diff --git a/_module/ncs/m_elfpermale.ncs b/_module/ncs/m_elfpermale.ncs index 74011b2..7890310 100644 Binary files a/_module/ncs/m_elfpermale.ncs and b/_module/ncs/m_elfpermale.ncs differ diff --git a/_module/ncs/mac_elfperfemale.ncs b/_module/ncs/mac_elfperfemale.ncs index efebdcd..7da13e8 100644 Binary files a/_module/ncs/mac_elfperfemale.ncs and b/_module/ncs/mac_elfperfemale.ncs differ diff --git a/_module/ncs/mac_elfpermale2.ncs b/_module/ncs/mac_elfpermale2.ncs index e170e32..297f36c 100644 Binary files a/_module/ncs/mac_elfpermale2.ncs and b/_module/ncs/mac_elfpermale2.ncs differ diff --git a/_module/ncs/mac_elftakehead.ncs b/_module/ncs/mac_elftakehead.ncs index 41daa59..eedac82 100644 Binary files a/_module/ncs/mac_elftakehead.ncs and b/_module/ncs/mac_elftakehead.ncs differ diff --git a/_module/ncs/mac_ladderport.ncs b/_module/ncs/mac_ladderport.ncs index a7dae3b..2a71189 100644 Binary files a/_module/ncs/mac_ladderport.ncs and b/_module/ncs/mac_ladderport.ncs differ diff --git a/_module/ncs/mac_orcheadcheck.ncs b/_module/ncs/mac_orcheadcheck.ncs index ed87921..a7c0d68 100644 Binary files a/_module/ncs/mac_orcheadcheck.ncs and b/_module/ncs/mac_orcheadcheck.ncs differ diff --git a/_module/ncs/mac_orcportal.ncs b/_module/ncs/mac_orcportal.ncs index ca3c863..a189777 100644 Binary files a/_module/ncs/mac_orcportal.ncs and b/_module/ncs/mac_orcportal.ncs differ diff --git a/_module/ncs/mac_wellport.ncs b/_module/ncs/mac_wellport.ncs index 6cdca8d..e5bb833 100644 Binary files a/_module/ncs/mac_wellport.ncs and b/_module/ncs/mac_wellport.ncs differ diff --git a/_module/ncs/mac_zeddsmartpor.ncs b/_module/ncs/mac_zeddsmartpor.ncs index a043837..699706f 100644 Binary files a/_module/ncs/mac_zeddsmartpor.ncs and b/_module/ncs/mac_zeddsmartpor.ncs differ diff --git a/_module/ncs/magering_gen.ncs b/_module/ncs/magering_gen.ncs index a10bf38..c555f5d 100644 Binary files a/_module/ncs/magering_gen.ncs and b/_module/ncs/magering_gen.ncs differ diff --git a/_module/ncs/mags2_port.ncs b/_module/ncs/mags2_port.ncs index 2c8b2b1..b6d2f24 100644 Binary files a/_module/ncs/mags2_port.ncs and b/_module/ncs/mags2_port.ncs differ diff --git a/_module/ncs/mags_south_port.ncs b/_module/ncs/mags_south_port.ncs index d347275..84a4991 100644 Binary files a/_module/ncs/mags_south_port.ncs and b/_module/ncs/mags_south_port.ncs differ diff --git a/_module/ncs/mared_head_check.ncs b/_module/ncs/mared_head_check.ncs index 5a45bfa..bfe15d1 100644 Binary files a/_module/ncs/mared_head_check.ncs and b/_module/ncs/mared_head_check.ncs differ diff --git a/_module/ncs/mared_head_take.ncs b/_module/ncs/mared_head_take.ncs index 4ed8b08..e816ec2 100644 Binary files a/_module/ncs/mared_head_take.ncs and b/_module/ncs/mared_head_take.ncs differ diff --git a/_module/ncs/mared_ring_chk.ncs b/_module/ncs/mared_ring_chk.ncs index 270190a..2cccfed 100644 Binary files a/_module/ncs/mared_ring_chk.ncs and b/_module/ncs/mared_ring_chk.ncs differ diff --git a/_module/ncs/mayor_give_gold.ncs b/_module/ncs/mayor_give_gold.ncs index c45cd1f..6ab2b15 100644 Binary files a/_module/ncs/mayor_give_gold.ncs and b/_module/ncs/mayor_give_gold.ncs differ diff --git a/_module/ncs/mayor_item_check.ncs b/_module/ncs/mayor_item_check.ncs index 556ded9..35e860b 100644 Binary files a/_module/ncs/mayor_item_check.ncs and b/_module/ncs/mayor_item_check.ncs differ diff --git a/_module/ncs/mayor_item_chk1.ncs b/_module/ncs/mayor_item_chk1.ncs index ccf6a8f..aecc58e 100644 Binary files a/_module/ncs/mayor_item_chk1.ncs and b/_module/ncs/mayor_item_chk1.ncs differ diff --git a/_module/ncs/mayor_item_chk2.ncs b/_module/ncs/mayor_item_chk2.ncs index 1c84967..fe38df9 100644 Binary files a/_module/ncs/mayor_item_chk2.ncs and b/_module/ncs/mayor_item_chk2.ncs differ diff --git a/_module/ncs/mayor_item_chk3.ncs b/_module/ncs/mayor_item_chk3.ncs index e0b50f1..dd187a1 100644 Binary files a/_module/ncs/mayor_item_chk3.ncs and b/_module/ncs/mayor_item_chk3.ncs differ diff --git a/_module/ncs/mayor_item_take.ncs b/_module/ncs/mayor_item_take.ncs index 95ce494..c7ec4eb 100644 Binary files a/_module/ncs/mayor_item_take.ncs and b/_module/ncs/mayor_item_take.ncs differ diff --git a/_module/ncs/mayor_take_lttr.ncs b/_module/ncs/mayor_take_lttr.ncs index ff212b9..31946aa 100644 Binary files a/_module/ncs/mayor_take_lttr.ncs and b/_module/ncs/mayor_take_lttr.ncs differ diff --git a/_module/ncs/mayor_var_chk.ncs b/_module/ncs/mayor_var_chk.ncs index 7524d3b..df5908c 100644 Binary files a/_module/ncs/mayor_var_chk.ncs and b/_module/ncs/mayor_var_chk.ncs differ diff --git a/_module/ncs/mc_jaganghead.ncs b/_module/ncs/mc_jaganghead.ncs index 770efc9..d41198a 100644 Binary files a/_module/ncs/mc_jaganghead.ncs and b/_module/ncs/mc_jaganghead.ncs differ diff --git a/_module/ncs/mcain_on_death.ncs b/_module/ncs/mcain_on_death.ncs index c94042f..1f0d6ea 100644 Binary files a/_module/ncs/mcain_on_death.ncs and b/_module/ncs/mcain_on_death.ncs differ diff --git a/_module/ncs/midlandringkey.ncs b/_module/ncs/midlandringkey.ncs index 03945c2..2d85bee 100644 Binary files a/_module/ncs/midlandringkey.ncs and b/_module/ncs/midlandringkey.ncs differ diff --git a/_module/ncs/mindport.ncs b/_module/ncs/mindport.ncs index e005e42..304084b 100644 Binary files a/_module/ncs/mindport.ncs and b/_module/ncs/mindport.ncs differ diff --git a/_module/ncs/mod_events.ncs b/_module/ncs/mod_events.ncs new file mode 100644 index 0000000..332bdf3 Binary files /dev/null and b/_module/ncs/mod_events.ncs differ diff --git a/_module/ncs/mod_nuievent.ncs b/_module/ncs/mod_nuievent.ncs new file mode 100644 index 0000000..380569f Binary files /dev/null and b/_module/ncs/mod_nuievent.ncs differ diff --git a/_module/ncs/mod_onplaytarget.ncs b/_module/ncs/mod_onplaytarget.ncs new file mode 100644 index 0000000..e3f4278 Binary files /dev/null and b/_module/ncs/mod_onplaytarget.ncs differ diff --git a/_module/ncs/monk_attack.ncs b/_module/ncs/monk_attack.ncs index 7bc1eae..d38cbda 100644 Binary files a/_module/ncs/monk_attack.ncs and b/_module/ncs/monk_attack.ncs differ diff --git a/_module/ncs/monk_port_back.ncs b/_module/ncs/monk_port_back.ncs index 2cff524..79675b1 100644 Binary files a/_module/ncs/monk_port_back.ncs and b/_module/ncs/monk_port_back.ncs differ diff --git a/_module/ncs/nui_f_storage.ncs b/_module/ncs/nui_f_storage.ncs new file mode 100644 index 0000000..24b672b Binary files /dev/null and b/_module/ncs/nui_f_storage.ncs differ diff --git a/_module/ncs/nw_c2_bossdie.ncs b/_module/ncs/nw_c2_bossdie.ncs index cf3a7d6..832c83b 100644 Binary files a/_module/ncs/nw_c2_bossdie.ncs and b/_module/ncs/nw_c2_bossdie.ncs differ diff --git a/_module/ncs/nw_c2_bossspawn.ncs b/_module/ncs/nw_c2_bossspawn.ncs index 7330f43..4445daf 100644 Binary files a/_module/ncs/nw_c2_bossspawn.ncs and b/_module/ncs/nw_c2_bossspawn.ncs differ diff --git a/_module/ncs/nw_c2_default1.ncs b/_module/ncs/nw_c2_default1.ncs index 4f5b1f9..5f191ed 100644 Binary files a/_module/ncs/nw_c2_default1.ncs and b/_module/ncs/nw_c2_default1.ncs differ diff --git a/_module/ncs/nw_c2_default2.ncs b/_module/ncs/nw_c2_default2.ncs index c3a4f7e..b8cf196 100644 Binary files a/_module/ncs/nw_c2_default2.ncs and b/_module/ncs/nw_c2_default2.ncs differ diff --git a/_module/ncs/nw_c2_default3.ncs b/_module/ncs/nw_c2_default3.ncs index c54384c..9ae6cba 100644 Binary files a/_module/ncs/nw_c2_default3.ncs and b/_module/ncs/nw_c2_default3.ncs differ diff --git a/_module/ncs/nw_c2_default4.ncs b/_module/ncs/nw_c2_default4.ncs index b8652e3..a465034 100644 Binary files a/_module/ncs/nw_c2_default4.ncs and b/_module/ncs/nw_c2_default4.ncs differ diff --git a/_module/ncs/nw_c2_default5.ncs b/_module/ncs/nw_c2_default5.ncs index 1502fbc..5fab86a 100644 Binary files a/_module/ncs/nw_c2_default5.ncs and b/_module/ncs/nw_c2_default5.ncs differ diff --git a/_module/ncs/nw_c2_default6.ncs b/_module/ncs/nw_c2_default6.ncs index 1727698..443cb4e 100644 Binary files a/_module/ncs/nw_c2_default6.ncs and b/_module/ncs/nw_c2_default6.ncs differ diff --git a/_module/ncs/nw_c2_default8.ncs b/_module/ncs/nw_c2_default8.ncs index 2df2091..a49b1b8 100644 Binary files a/_module/ncs/nw_c2_default8.ncs and b/_module/ncs/nw_c2_default8.ncs differ diff --git a/_module/ncs/nw_c2_defaultb.ncs b/_module/ncs/nw_c2_defaultb.ncs index 146034f..c03dfc4 100644 Binary files a/_module/ncs/nw_c2_defaultb.ncs and b/_module/ncs/nw_c2_defaultb.ncs differ diff --git a/_module/ncs/nw_c2_defaulte.ncs b/_module/ncs/nw_c2_defaulte.ncs index 1eb9873..b95ac13 100644 Binary files a/_module/ncs/nw_c2_defaulte.ncs and b/_module/ncs/nw_c2_defaulte.ncs differ diff --git a/_module/ncs/nw_d2_gen_check.ncs b/_module/ncs/nw_d2_gen_check.ncs index 847c96c..84a51c9 100644 Binary files a/_module/ncs/nw_d2_gen_check.ncs and b/_module/ncs/nw_d2_gen_check.ncs differ diff --git a/_module/ncs/nw_o2_boss.ncs b/_module/ncs/nw_o2_boss.ncs index c11a213..11cf573 100644 Binary files a/_module/ncs/nw_o2_boss.ncs and b/_module/ncs/nw_o2_boss.ncs differ diff --git a/_module/ncs/nw_o2_classweap.ncs b/_module/ncs/nw_o2_classweap.ncs index ff79262..bb5ae9a 100644 Binary files a/_module/ncs/nw_o2_classweap.ncs and b/_module/ncs/nw_o2_classweap.ncs differ diff --git a/_module/ncs/nw_o2_generalhig.ncs b/_module/ncs/nw_o2_generalhig.ncs index 56d89fc..3dbda6b 100644 Binary files a/_module/ncs/nw_o2_generalhig.ncs and b/_module/ncs/nw_o2_generalhig.ncs differ diff --git a/_module/ncs/nw_o2_generallow.ncs b/_module/ncs/nw_o2_generallow.ncs index d435f66..304d23d 100644 Binary files a/_module/ncs/nw_o2_generallow.ncs and b/_module/ncs/nw_o2_generallow.ncs differ diff --git a/_module/ncs/nw_o2_generalmed.ncs b/_module/ncs/nw_o2_generalmed.ncs index 01f40f4..9bf41ec 100644 Binary files a/_module/ncs/nw_o2_generalmed.ncs and b/_module/ncs/nw_o2_generalmed.ncs differ diff --git a/_module/ncs/nw_s1_aurablnda.ncs b/_module/ncs/nw_s1_aurablnda.ncs index e4aa3c2..12d209f 100644 Binary files a/_module/ncs/nw_s1_aurablnda.ncs and b/_module/ncs/nw_s1_aurablnda.ncs differ diff --git a/_module/ncs/nw_s1_auracoldc.ncs b/_module/ncs/nw_s1_auracoldc.ncs index a5e7712..f48acae 100644 Binary files a/_module/ncs/nw_s1_auracoldc.ncs and b/_module/ncs/nw_s1_auracoldc.ncs differ diff --git a/_module/ncs/nw_s1_auraelecc.ncs b/_module/ncs/nw_s1_auraelecc.ncs index 2d8e4c0..d5c53e8 100644 Binary files a/_module/ncs/nw_s1_auraelecc.ncs and b/_module/ncs/nw_s1_auraelecc.ncs differ diff --git a/_module/ncs/nw_s1_aurafirec.ncs b/_module/ncs/nw_s1_aurafirec.ncs index 725e169..698a00d 100644 Binary files a/_module/ncs/nw_s1_aurafirec.ncs and b/_module/ncs/nw_s1_aurafirec.ncs differ diff --git a/_module/ncs/nw_s1_auramenca.ncs b/_module/ncs/nw_s1_auramenca.ncs index 507f037..376d33b 100644 Binary files a/_module/ncs/nw_s1_auramenca.ncs and b/_module/ncs/nw_s1_auramenca.ncs differ diff --git a/_module/ncs/nw_s1_auraprota.ncs b/_module/ncs/nw_s1_auraprota.ncs index cea1eec..b4f9d23 100644 Binary files a/_module/ncs/nw_s1_auraprota.ncs and b/_module/ncs/nw_s1_auraprota.ncs differ diff --git a/_module/ncs/nw_s1_aurastuna.ncs b/_module/ncs/nw_s1_aurastuna.ncs index b6eda58..c445df2 100644 Binary files a/_module/ncs/nw_s1_aurastuna.ncs and b/_module/ncs/nw_s1_aurastuna.ncs differ diff --git a/_module/ncs/nw_s1_auraunata.ncs b/_module/ncs/nw_s1_auraunata.ncs index 8ebd947..0224609 100644 Binary files a/_module/ncs/nw_s1_auraunata.ncs and b/_module/ncs/nw_s1_auraunata.ncs differ diff --git a/_module/ncs/nw_s1_aurauneaa.ncs b/_module/ncs/nw_s1_aurauneaa.ncs index 34bc167..2b0ef5c 100644 Binary files a/_module/ncs/nw_s1_aurauneaa.ncs and b/_module/ncs/nw_s1_aurauneaa.ncs differ diff --git a/_module/ncs/nw_s1_bltacid.ncs b/_module/ncs/nw_s1_bltacid.ncs index a5952b3..ce54781 100644 Binary files a/_module/ncs/nw_s1_bltacid.ncs and b/_module/ncs/nw_s1_bltacid.ncs differ diff --git a/_module/ncs/nw_s1_bltcharm.ncs b/_module/ncs/nw_s1_bltcharm.ncs index 2a37c20..3758046 100644 Binary files a/_module/ncs/nw_s1_bltcharm.ncs and b/_module/ncs/nw_s1_bltcharm.ncs differ diff --git a/_module/ncs/nw_s1_bltchrdr.ncs b/_module/ncs/nw_s1_bltchrdr.ncs index 1d69902..41f50cb 100644 Binary files a/_module/ncs/nw_s1_bltchrdr.ncs and b/_module/ncs/nw_s1_bltchrdr.ncs differ diff --git a/_module/ncs/nw_s1_bltcold.ncs b/_module/ncs/nw_s1_bltcold.ncs index 2d00a86..11c3735 100644 Binary files a/_module/ncs/nw_s1_bltcold.ncs and b/_module/ncs/nw_s1_bltcold.ncs differ diff --git a/_module/ncs/nw_s1_bltcondr.ncs b/_module/ncs/nw_s1_bltcondr.ncs index 6560555..35c2bda 100644 Binary files a/_module/ncs/nw_s1_bltcondr.ncs and b/_module/ncs/nw_s1_bltcondr.ncs differ diff --git a/_module/ncs/nw_s1_bltconf.ncs b/_module/ncs/nw_s1_bltconf.ncs index e74ecf4..da81d8c 100644 Binary files a/_module/ncs/nw_s1_bltconf.ncs and b/_module/ncs/nw_s1_bltconf.ncs differ diff --git a/_module/ncs/nw_s1_bltdaze.ncs b/_module/ncs/nw_s1_bltdaze.ncs index b8ec5d2..48b4d76 100644 Binary files a/_module/ncs/nw_s1_bltdaze.ncs and b/_module/ncs/nw_s1_bltdaze.ncs differ diff --git a/_module/ncs/nw_s1_bltdeath.ncs b/_module/ncs/nw_s1_bltdeath.ncs index ef10b59..9aecda2 100644 Binary files a/_module/ncs/nw_s1_bltdeath.ncs and b/_module/ncs/nw_s1_bltdeath.ncs differ diff --git a/_module/ncs/nw_s1_bltdexdr.ncs b/_module/ncs/nw_s1_bltdexdr.ncs index 2904c69..137f50b 100644 Binary files a/_module/ncs/nw_s1_bltdexdr.ncs and b/_module/ncs/nw_s1_bltdexdr.ncs differ diff --git a/_module/ncs/nw_s1_bltdisese.ncs b/_module/ncs/nw_s1_bltdisese.ncs index c81e0cc..384e807 100644 Binary files a/_module/ncs/nw_s1_bltdisese.ncs and b/_module/ncs/nw_s1_bltdisese.ncs differ diff --git a/_module/ncs/nw_s1_bltdomn.ncs b/_module/ncs/nw_s1_bltdomn.ncs index 3c354b3..9a19912 100644 Binary files a/_module/ncs/nw_s1_bltdomn.ncs and b/_module/ncs/nw_s1_bltdomn.ncs differ diff --git a/_module/ncs/nw_s1_bltfire.ncs b/_module/ncs/nw_s1_bltfire.ncs index 09ac47c..ac48c2e 100644 Binary files a/_module/ncs/nw_s1_bltfire.ncs and b/_module/ncs/nw_s1_bltfire.ncs differ diff --git a/_module/ncs/nw_s1_bltintdr.ncs b/_module/ncs/nw_s1_bltintdr.ncs index 748e670..6a455a6 100644 Binary files a/_module/ncs/nw_s1_bltintdr.ncs and b/_module/ncs/nw_s1_bltintdr.ncs differ diff --git a/_module/ncs/nw_s1_bltknckd.ncs b/_module/ncs/nw_s1_bltknckd.ncs index e9a8d53..03c8479 100644 Binary files a/_module/ncs/nw_s1_bltknckd.ncs and b/_module/ncs/nw_s1_bltknckd.ncs differ diff --git a/_module/ncs/nw_s1_bltlightn.ncs b/_module/ncs/nw_s1_bltlightn.ncs index 0286bec..c19125b 100644 Binary files a/_module/ncs/nw_s1_bltlightn.ncs and b/_module/ncs/nw_s1_bltlightn.ncs differ diff --git a/_module/ncs/nw_s1_bltlvldr.ncs b/_module/ncs/nw_s1_bltlvldr.ncs index e8628c5..2c6a5ff 100644 Binary files a/_module/ncs/nw_s1_bltlvldr.ncs and b/_module/ncs/nw_s1_bltlvldr.ncs differ diff --git a/_module/ncs/nw_s1_bltparal.ncs b/_module/ncs/nw_s1_bltparal.ncs index 95cce17..036ac71 100644 Binary files a/_module/ncs/nw_s1_bltparal.ncs and b/_module/ncs/nw_s1_bltparal.ncs differ diff --git a/_module/ncs/nw_s1_bltpoison.ncs b/_module/ncs/nw_s1_bltpoison.ncs index e4066f7..ef49a4e 100644 Binary files a/_module/ncs/nw_s1_bltpoison.ncs and b/_module/ncs/nw_s1_bltpoison.ncs differ diff --git a/_module/ncs/nw_s1_bltshards.ncs b/_module/ncs/nw_s1_bltshards.ncs index 00a5432..742d509 100644 Binary files a/_module/ncs/nw_s1_bltshards.ncs and b/_module/ncs/nw_s1_bltshards.ncs differ diff --git a/_module/ncs/nw_s1_bltslow.ncs b/_module/ncs/nw_s1_bltslow.ncs index d59eaa3..ca772b2 100644 Binary files a/_module/ncs/nw_s1_bltslow.ncs and b/_module/ncs/nw_s1_bltslow.ncs differ diff --git a/_module/ncs/nw_s1_bltstrdr.ncs b/_module/ncs/nw_s1_bltstrdr.ncs index 7819b3e..8f8e218 100644 Binary files a/_module/ncs/nw_s1_bltstrdr.ncs and b/_module/ncs/nw_s1_bltstrdr.ncs differ diff --git a/_module/ncs/nw_s1_bltstun.ncs b/_module/ncs/nw_s1_bltstun.ncs index 7a8f0d3..f3de7fb 100644 Binary files a/_module/ncs/nw_s1_bltstun.ncs and b/_module/ncs/nw_s1_bltstun.ncs differ diff --git a/_module/ncs/nw_s1_bltweb.ncs b/_module/ncs/nw_s1_bltweb.ncs index 67c3671..fa6920b 100644 Binary files a/_module/ncs/nw_s1_bltweb.ncs and b/_module/ncs/nw_s1_bltweb.ncs differ diff --git a/_module/ncs/nw_s1_bltwisdr.ncs b/_module/ncs/nw_s1_bltwisdr.ncs index 93d66dc..f460a1a 100644 Binary files a/_module/ncs/nw_s1_bltwisdr.ncs and b/_module/ncs/nw_s1_bltwisdr.ncs differ diff --git a/_module/ncs/nw_s1_coneacid.ncs b/_module/ncs/nw_s1_coneacid.ncs index 9b65127..35a041f 100644 Binary files a/_module/ncs/nw_s1_coneacid.ncs and b/_module/ncs/nw_s1_coneacid.ncs differ diff --git a/_module/ncs/nw_s1_conecold.ncs b/_module/ncs/nw_s1_conecold.ncs index 4f75c6e..3295fe0 100644 Binary files a/_module/ncs/nw_s1_conecold.ncs and b/_module/ncs/nw_s1_conecold.ncs differ diff --git a/_module/ncs/nw_s1_conedisea.ncs b/_module/ncs/nw_s1_conedisea.ncs index 7ddb012..d2a8cb5 100644 Binary files a/_module/ncs/nw_s1_conedisea.ncs and b/_module/ncs/nw_s1_conedisea.ncs differ diff --git a/_module/ncs/nw_s1_coneelec.ncs b/_module/ncs/nw_s1_coneelec.ncs index 11ecb35..0bd2a68 100644 Binary files a/_module/ncs/nw_s1_coneelec.ncs and b/_module/ncs/nw_s1_coneelec.ncs differ diff --git a/_module/ncs/nw_s1_conesonic.ncs b/_module/ncs/nw_s1_conesonic.ncs index c6a3027..1d2c229 100644 Binary files a/_module/ncs/nw_s1_conesonic.ncs and b/_module/ncs/nw_s1_conesonic.ncs differ diff --git a/_module/ncs/nw_s1_dragfear.ncs b/_module/ncs/nw_s1_dragfear.ncs index 5fd6d0a..f6edb14 100644 Binary files a/_module/ncs/nw_s1_dragfear.ncs and b/_module/ncs/nw_s1_dragfear.ncs differ diff --git a/_module/ncs/nw_s1_dragfeara.ncs b/_module/ncs/nw_s1_dragfeara.ncs index fe3245d..c7eca68 100644 Binary files a/_module/ncs/nw_s1_dragfeara.ncs and b/_module/ncs/nw_s1_dragfeara.ncs differ diff --git a/_module/ncs/nw_s1_feroc3.ncs b/_module/ncs/nw_s1_feroc3.ncs index dfb32e0..ef2abee 100644 Binary files a/_module/ncs/nw_s1_feroc3.ncs and b/_module/ncs/nw_s1_feroc3.ncs differ diff --git a/_module/ncs/nw_s1_gazechaos.ncs b/_module/ncs/nw_s1_gazechaos.ncs index 5f62c74..0b1e1ef 100644 Binary files a/_module/ncs/nw_s1_gazechaos.ncs and b/_module/ncs/nw_s1_gazechaos.ncs differ diff --git a/_module/ncs/nw_s1_gazecharm.ncs b/_module/ncs/nw_s1_gazecharm.ncs index f5b734c..0cc7226 100644 Binary files a/_module/ncs/nw_s1_gazecharm.ncs and b/_module/ncs/nw_s1_gazecharm.ncs differ diff --git a/_module/ncs/nw_s1_gazeconfu.ncs b/_module/ncs/nw_s1_gazeconfu.ncs index 58011c9..c805ec1 100644 Binary files a/_module/ncs/nw_s1_gazeconfu.ncs and b/_module/ncs/nw_s1_gazeconfu.ncs differ diff --git a/_module/ncs/nw_s1_gazedaze.ncs b/_module/ncs/nw_s1_gazedaze.ncs index 340d88c..817e0ca 100644 Binary files a/_module/ncs/nw_s1_gazedaze.ncs and b/_module/ncs/nw_s1_gazedaze.ncs differ diff --git a/_module/ncs/nw_s1_gazedeath.ncs b/_module/ncs/nw_s1_gazedeath.ncs index 8c6da0f..963cd69 100644 Binary files a/_module/ncs/nw_s1_gazedeath.ncs and b/_module/ncs/nw_s1_gazedeath.ncs differ diff --git a/_module/ncs/nw_s1_gazedomn.ncs b/_module/ncs/nw_s1_gazedomn.ncs index abbfe06..a36186d 100644 Binary files a/_module/ncs/nw_s1_gazedomn.ncs and b/_module/ncs/nw_s1_gazedomn.ncs differ diff --git a/_module/ncs/nw_s1_gazedoom.ncs b/_module/ncs/nw_s1_gazedoom.ncs index 459bd71..92a28cc 100644 Binary files a/_module/ncs/nw_s1_gazedoom.ncs and b/_module/ncs/nw_s1_gazedoom.ncs differ diff --git a/_module/ncs/nw_s1_gazeevil.ncs b/_module/ncs/nw_s1_gazeevil.ncs index e59ef93..b3d0eb6 100644 Binary files a/_module/ncs/nw_s1_gazeevil.ncs and b/_module/ncs/nw_s1_gazeevil.ncs differ diff --git a/_module/ncs/nw_s1_gazefear.ncs b/_module/ncs/nw_s1_gazefear.ncs index 1933b6a..0b0a58f 100644 Binary files a/_module/ncs/nw_s1_gazefear.ncs and b/_module/ncs/nw_s1_gazefear.ncs differ diff --git a/_module/ncs/nw_s1_gazegood.ncs b/_module/ncs/nw_s1_gazegood.ncs index 22b27ce..cb3e19e 100644 Binary files a/_module/ncs/nw_s1_gazegood.ncs and b/_module/ncs/nw_s1_gazegood.ncs differ diff --git a/_module/ncs/nw_s1_gazelaw.ncs b/_module/ncs/nw_s1_gazelaw.ncs index 9869323..ad6753c 100644 Binary files a/_module/ncs/nw_s1_gazelaw.ncs and b/_module/ncs/nw_s1_gazelaw.ncs differ diff --git a/_module/ncs/nw_s1_gazestun.ncs b/_module/ncs/nw_s1_gazestun.ncs index d6fc354..fa5efee 100644 Binary files a/_module/ncs/nw_s1_gazestun.ncs and b/_module/ncs/nw_s1_gazestun.ncs differ diff --git a/_module/ncs/nw_s1_golemgas.ncs b/_module/ncs/nw_s1_golemgas.ncs index efb74ee..c54e9ec 100644 Binary files a/_module/ncs/nw_s1_golemgas.ncs and b/_module/ncs/nw_s1_golemgas.ncs differ diff --git a/_module/ncs/nw_s1_hndbreath.ncs b/_module/ncs/nw_s1_hndbreath.ncs index 39b6ab6..f05372a 100644 Binary files a/_module/ncs/nw_s1_hndbreath.ncs and b/_module/ncs/nw_s1_hndbreath.ncs differ diff --git a/_module/ncs/nw_s1_howlconf.ncs b/_module/ncs/nw_s1_howlconf.ncs index 2649dad..3128bf2 100644 Binary files a/_module/ncs/nw_s1_howlconf.ncs and b/_module/ncs/nw_s1_howlconf.ncs differ diff --git a/_module/ncs/nw_s1_howldaze.ncs b/_module/ncs/nw_s1_howldaze.ncs index fc595ff..a27eec2 100644 Binary files a/_module/ncs/nw_s1_howldaze.ncs and b/_module/ncs/nw_s1_howldaze.ncs differ diff --git a/_module/ncs/nw_s1_howldeath.ncs b/_module/ncs/nw_s1_howldeath.ncs index c1f8a98..5259895 100644 Binary files a/_module/ncs/nw_s1_howldeath.ncs and b/_module/ncs/nw_s1_howldeath.ncs differ diff --git a/_module/ncs/nw_s1_howlfear.ncs b/_module/ncs/nw_s1_howlfear.ncs index 1ac9f0e..7f9eb6d 100644 Binary files a/_module/ncs/nw_s1_howlfear.ncs and b/_module/ncs/nw_s1_howlfear.ncs differ diff --git a/_module/ncs/nw_s1_howlparal.ncs b/_module/ncs/nw_s1_howlparal.ncs index 42488a9..5dffdc3 100644 Binary files a/_module/ncs/nw_s1_howlparal.ncs and b/_module/ncs/nw_s1_howlparal.ncs differ diff --git a/_module/ncs/nw_s1_howlsonic.ncs b/_module/ncs/nw_s1_howlsonic.ncs index ae721fd..b9b3f2d 100644 Binary files a/_module/ncs/nw_s1_howlsonic.ncs and b/_module/ncs/nw_s1_howlsonic.ncs differ diff --git a/_module/ncs/nw_s1_howlstun.ncs b/_module/ncs/nw_s1_howlstun.ncs index 9a8d7f6..fcf0557 100644 Binary files a/_module/ncs/nw_s1_howlstun.ncs and b/_module/ncs/nw_s1_howlstun.ncs differ diff --git a/_module/ncs/nw_s1_krenscare.ncs b/_module/ncs/nw_s1_krenscare.ncs index 8f7fa04..96d2acb 100644 Binary files a/_module/ncs/nw_s1_krenscare.ncs and b/_module/ncs/nw_s1_krenscare.ncs differ diff --git a/_module/ncs/nw_s1_mephsalt.ncs b/_module/ncs/nw_s1_mephsalt.ncs index 5406ee0..685a8f9 100644 Binary files a/_module/ncs/nw_s1_mephsalt.ncs and b/_module/ncs/nw_s1_mephsalt.ncs differ diff --git a/_module/ncs/nw_s1_mephsteam.ncs b/_module/ncs/nw_s1_mephsteam.ncs index f61d7ef..b7d9512 100644 Binary files a/_module/ncs/nw_s1_mephsteam.ncs and b/_module/ncs/nw_s1_mephsteam.ncs differ diff --git a/_module/ncs/nw_s1_mumundead.ncs b/_module/ncs/nw_s1_mumundead.ncs index 2205868..c2e4820 100644 Binary files a/_module/ncs/nw_s1_mumundead.ncs and b/_module/ncs/nw_s1_mumundead.ncs differ diff --git a/_module/ncs/nw_s1_pulschrdr.ncs b/_module/ncs/nw_s1_pulschrdr.ncs index acc6c11..e0de12a 100644 Binary files a/_module/ncs/nw_s1_pulschrdr.ncs and b/_module/ncs/nw_s1_pulschrdr.ncs differ diff --git a/_module/ncs/nw_s1_pulscold.ncs b/_module/ncs/nw_s1_pulscold.ncs index 23df913..958973b 100644 Binary files a/_module/ncs/nw_s1_pulscold.ncs and b/_module/ncs/nw_s1_pulscold.ncs differ diff --git a/_module/ncs/nw_s1_pulscondr.ncs b/_module/ncs/nw_s1_pulscondr.ncs index 39b62d7..db7486a 100644 Binary files a/_module/ncs/nw_s1_pulscondr.ncs and b/_module/ncs/nw_s1_pulscondr.ncs differ diff --git a/_module/ncs/nw_s1_pulsdeath.ncs b/_module/ncs/nw_s1_pulsdeath.ncs index 6879326..7a55dcf 100644 Binary files a/_module/ncs/nw_s1_pulsdeath.ncs and b/_module/ncs/nw_s1_pulsdeath.ncs differ diff --git a/_module/ncs/nw_s1_pulsdexdr.ncs b/_module/ncs/nw_s1_pulsdexdr.ncs index 60b1f47..fa1a63f 100644 Binary files a/_module/ncs/nw_s1_pulsdexdr.ncs and b/_module/ncs/nw_s1_pulsdexdr.ncs differ diff --git a/_module/ncs/nw_s1_pulsdis.ncs b/_module/ncs/nw_s1_pulsdis.ncs index 39c19ba..35da2a7 100644 Binary files a/_module/ncs/nw_s1_pulsdis.ncs and b/_module/ncs/nw_s1_pulsdis.ncs differ diff --git a/_module/ncs/nw_s1_pulselec.ncs b/_module/ncs/nw_s1_pulselec.ncs index dd6a5f0..b9b061f 100644 Binary files a/_module/ncs/nw_s1_pulselec.ncs and b/_module/ncs/nw_s1_pulselec.ncs differ diff --git a/_module/ncs/nw_s1_pulsfire.ncs b/_module/ncs/nw_s1_pulsfire.ncs index 1da13e5..3bb40b3 100644 Binary files a/_module/ncs/nw_s1_pulsfire.ncs and b/_module/ncs/nw_s1_pulsfire.ncs differ diff --git a/_module/ncs/nw_s1_pulsholy.ncs b/_module/ncs/nw_s1_pulsholy.ncs index 3b93769..5d1e4fd 100644 Binary files a/_module/ncs/nw_s1_pulsholy.ncs and b/_module/ncs/nw_s1_pulsholy.ncs differ diff --git a/_module/ncs/nw_s1_pulsintdr.ncs b/_module/ncs/nw_s1_pulsintdr.ncs index ddf504a..67909a5 100644 Binary files a/_module/ncs/nw_s1_pulsintdr.ncs and b/_module/ncs/nw_s1_pulsintdr.ncs differ diff --git a/_module/ncs/nw_s1_pulslvldr.ncs b/_module/ncs/nw_s1_pulslvldr.ncs index 408bf6e..9e27953 100644 Binary files a/_module/ncs/nw_s1_pulslvldr.ncs and b/_module/ncs/nw_s1_pulslvldr.ncs differ diff --git a/_module/ncs/nw_s1_pulsneg.ncs b/_module/ncs/nw_s1_pulsneg.ncs index 7dc823b..cd01606 100644 Binary files a/_module/ncs/nw_s1_pulsneg.ncs and b/_module/ncs/nw_s1_pulsneg.ncs differ diff --git a/_module/ncs/nw_s1_pulspois.ncs b/_module/ncs/nw_s1_pulspois.ncs index a381a67..1baac1c 100644 Binary files a/_module/ncs/nw_s1_pulspois.ncs and b/_module/ncs/nw_s1_pulspois.ncs differ diff --git a/_module/ncs/nw_s1_pulsspore.ncs b/_module/ncs/nw_s1_pulsspore.ncs index 4f7e590..ca31fbe 100644 Binary files a/_module/ncs/nw_s1_pulsspore.ncs and b/_module/ncs/nw_s1_pulsspore.ncs differ diff --git a/_module/ncs/nw_s1_pulsstrdr.ncs b/_module/ncs/nw_s1_pulsstrdr.ncs index 704c1ca..c8a8d6e 100644 Binary files a/_module/ncs/nw_s1_pulsstrdr.ncs and b/_module/ncs/nw_s1_pulsstrdr.ncs differ diff --git a/_module/ncs/nw_s1_pulswind.ncs b/_module/ncs/nw_s1_pulswind.ncs index 75d4133..25bebd2 100644 Binary files a/_module/ncs/nw_s1_pulswind.ncs and b/_module/ncs/nw_s1_pulswind.ncs differ diff --git a/_module/ncs/nw_s1_pulswisdr.ncs b/_module/ncs/nw_s1_pulswisdr.ncs index da829ca..4672aec 100644 Binary files a/_module/ncs/nw_s1_pulswisdr.ncs and b/_module/ncs/nw_s1_pulswisdr.ncs differ diff --git a/_module/ncs/nw_s1_smokeclaw.ncs b/_module/ncs/nw_s1_smokeclaw.ncs index 2e5a547..c523be6 100644 Binary files a/_module/ncs/nw_s1_smokeclaw.ncs and b/_module/ncs/nw_s1_smokeclaw.ncs differ diff --git a/_module/ncs/nw_s1_stink_a.ncs b/_module/ncs/nw_s1_stink_a.ncs index bca10c2..5d1c495 100644 Binary files a/_module/ncs/nw_s1_stink_a.ncs and b/_module/ncs/nw_s1_stink_a.ncs differ diff --git a/_module/ncs/nw_s1_tyrantfga.ncs b/_module/ncs/nw_s1_tyrantfga.ncs index 5f27b26..7104f7f 100644 Binary files a/_module/ncs/nw_s1_tyrantfga.ncs and b/_module/ncs/nw_s1_tyrantfga.ncs differ diff --git a/_module/ncs/nw_s1_tyrantfog.ncs b/_module/ncs/nw_s1_tyrantfog.ncs index 8ba90b5..908d8c3 100644 Binary files a/_module/ncs/nw_s1_tyrantfog.ncs and b/_module/ncs/nw_s1_tyrantfog.ncs differ diff --git a/_module/ncs/nw_s2_divprot.ncs b/_module/ncs/nw_s2_divprot.ncs index 7f7632c..9103a98 100644 Binary files a/_module/ncs/nw_s2_divprot.ncs and b/_module/ncs/nw_s2_divprot.ncs differ diff --git a/_module/ncs/nw_s3_balordeth.ncs b/_module/ncs/nw_s3_balordeth.ncs index 4e9dd36..1374b8a 100644 Binary files a/_module/ncs/nw_s3_balordeth.ncs and b/_module/ncs/nw_s3_balordeth.ncs differ diff --git a/_module/ncs/obj_hb_bubble.ncs b/_module/ncs/obj_hb_bubble.ncs index e74ccd9..1fe2aec 100644 Binary files a/_module/ncs/obj_hb_bubble.ncs and b/_module/ncs/obj_hb_bubble.ncs differ diff --git a/_module/ncs/omw_ppis_disturb.ncs b/_module/ncs/omw_ppis_disturb.ncs index 4f20922..682b5d7 100644 Binary files a/_module/ncs/omw_ppis_disturb.ncs and b/_module/ncs/omw_ppis_disturb.ncs differ diff --git a/_module/ncs/omw_ppis_start.ncs b/_module/ncs/omw_ppis_start.ncs index cd3a9dd..5f77c88 100644 Binary files a/_module/ncs/omw_ppis_start.ncs and b/_module/ncs/omw_ppis_start.ncs differ diff --git a/_module/ncs/open_field1.ncs b/_module/ncs/open_field1.ncs index 7f8697d..27ee025 100644 Binary files a/_module/ncs/open_field1.ncs and b/_module/ncs/open_field1.ncs differ diff --git a/_module/ncs/open_field2.ncs b/_module/ncs/open_field2.ncs index 1603fbc..0657943 100644 Binary files a/_module/ncs/open_field2.ncs and b/_module/ncs/open_field2.ncs differ diff --git a/_module/ncs/open_field3.ncs b/_module/ncs/open_field3.ncs index 5d886f3..59c669d 100644 Binary files a/_module/ncs/open_field3.ncs and b/_module/ncs/open_field3.ncs differ diff --git a/_module/ncs/open_field4.ncs b/_module/ncs/open_field4.ncs index f9039e7..9b02893 100644 Binary files a/_module/ncs/open_field4.ncs and b/_module/ncs/open_field4.ncs differ diff --git a/_module/ncs/open_field5.ncs b/_module/ncs/open_field5.ncs index 6cadf69..0ea443f 100644 Binary files a/_module/ncs/open_field5.ncs and b/_module/ncs/open_field5.ncs differ diff --git a/_module/ncs/open_field6.ncs b/_module/ncs/open_field6.ncs index 65ea99c..8c8a922 100644 Binary files a/_module/ncs/open_field6.ncs and b/_module/ncs/open_field6.ncs differ diff --git a/_module/ncs/pc_room_onexit.ncs b/_module/ncs/pc_room_onexit.ncs new file mode 100644 index 0000000..fb078ba Binary files /dev/null and b/_module/ncs/pc_room_onexit.ncs differ diff --git a/_module/ncs/perdition_forge.ncs b/_module/ncs/perdition_forge.ncs index e6c7ef9..ff2269e 100644 Binary files a/_module/ncs/perdition_forge.ncs and b/_module/ncs/perdition_forge.ncs differ diff --git a/_module/ncs/peruppi_clss_chk.ncs b/_module/ncs/peruppi_clss_chk.ncs index 1a5f2ac..3d5c87f 100644 Binary files a/_module/ncs/peruppi_clss_chk.ncs and b/_module/ncs/peruppi_clss_chk.ncs differ diff --git a/_module/ncs/prc_pwondeath.ncs b/_module/ncs/prc_pwondeath.ncs index 0531450..d48d2ce 100644 Binary files a/_module/ncs/prc_pwondeath.ncs and b/_module/ncs/prc_pwondeath.ncs differ diff --git a/_module/ncs/pvp_evil_flag.ncs b/_module/ncs/pvp_evil_flag.ncs index 6a27f89..7be716a 100644 Binary files a/_module/ncs/pvp_evil_flag.ncs and b/_module/ncs/pvp_evil_flag.ncs differ diff --git a/_module/ncs/pvp_giveevilorb.ncs b/_module/ncs/pvp_giveevilorb.ncs index a510eaa..27448ce 100644 Binary files a/_module/ncs/pvp_giveevilorb.ncs and b/_module/ncs/pvp_giveevilorb.ncs differ diff --git a/_module/ncs/pvp_givegoodorb.ncs b/_module/ncs/pvp_givegoodorb.ncs index 4901d35..baff054 100644 Binary files a/_module/ncs/pvp_givegoodorb.ncs and b/_module/ncs/pvp_givegoodorb.ncs differ diff --git a/_module/ncs/pvp_good_flag.ncs b/_module/ncs/pvp_good_flag.ncs index 471dea3..36d5a77 100644 Binary files a/_module/ncs/pvp_good_flag.ncs and b/_module/ncs/pvp_good_flag.ncs differ diff --git a/_module/ncs/pvp_returnevil.ncs b/_module/ncs/pvp_returnevil.ncs index 861b50b..4e0cacc 100644 Binary files a/_module/ncs/pvp_returnevil.ncs and b/_module/ncs/pvp_returnevil.ncs differ diff --git a/_module/ncs/pvp_returngood.ncs b/_module/ncs/pvp_returngood.ncs index 03440b3..08bf582 100644 Binary files a/_module/ncs/pvp_returngood.ncs and b/_module/ncs/pvp_returngood.ncs differ diff --git a/_module/ncs/pvp_tke_token.ncs b/_module/ncs/pvp_tke_token.ncs index 6cfc418..926f77c 100644 Binary files a/_module/ncs/pvp_tke_token.ncs and b/_module/ncs/pvp_tke_token.ncs differ diff --git a/_module/ncs/pws_onacquire.ncs b/_module/ncs/pws_onacquire.ncs index 33807c4..770006a 100644 Binary files a/_module/ncs/pws_onacquire.ncs and b/_module/ncs/pws_onacquire.ncs differ diff --git a/_module/ncs/pws_onunacquire.ncs b/_module/ncs/pws_onunacquire.ncs index b48a8c9..ee4ee42 100644 Binary files a/_module/ncs/pws_onunacquire.ncs and b/_module/ncs/pws_onunacquire.ncs differ diff --git a/_module/ncs/q2b_hb_pillar.ncs b/_module/ncs/q2b_hb_pillar.ncs index 9a14219..6691481 100644 Binary files a/_module/ncs/q2b_hb_pillar.ncs and b/_module/ncs/q2b_hb_pillar.ncs differ diff --git a/_module/ncs/quest_bay1.ncs b/_module/ncs/quest_bay1.ncs index 1fb8b19..22ffe47 100644 Binary files a/_module/ncs/quest_bay1.ncs and b/_module/ncs/quest_bay1.ncs differ diff --git a/_module/ncs/quest_bay2.ncs b/_module/ncs/quest_bay2.ncs index 1f3ff27..3c4d468 100644 Binary files a/_module/ncs/quest_bay2.ncs and b/_module/ncs/quest_bay2.ncs differ diff --git a/_module/ncs/quest_brew1.ncs b/_module/ncs/quest_brew1.ncs index c0f0735..996c09a 100644 Binary files a/_module/ncs/quest_brew1.ncs and b/_module/ncs/quest_brew1.ncs differ diff --git a/_module/ncs/quest_brew2.ncs b/_module/ncs/quest_brew2.ncs index b93093a..308825d 100644 Binary files a/_module/ncs/quest_brew2.ncs and b/_module/ncs/quest_brew2.ncs differ diff --git a/_module/ncs/quest_cloak1.ncs b/_module/ncs/quest_cloak1.ncs index a7edc1f..4af62be 100644 Binary files a/_module/ncs/quest_cloak1.ncs and b/_module/ncs/quest_cloak1.ncs differ diff --git a/_module/ncs/quest_cloak2.ncs b/_module/ncs/quest_cloak2.ncs index ebc2b67..e510e58 100644 Binary files a/_module/ncs/quest_cloak2.ncs and b/_module/ncs/quest_cloak2.ncs differ diff --git a/_module/ncs/quest_cloak3.ncs b/_module/ncs/quest_cloak3.ncs index 0dc844e..c0282c7 100644 Binary files a/_module/ncs/quest_cloak3.ncs and b/_module/ncs/quest_cloak3.ncs differ diff --git a/_module/ncs/quest_cloak4.ncs b/_module/ncs/quest_cloak4.ncs index cc98978..80afd90 100644 Binary files a/_module/ncs/quest_cloak4.ncs and b/_module/ncs/quest_cloak4.ncs differ diff --git a/_module/ncs/quest_cloak6.ncs b/_module/ncs/quest_cloak6.ncs index f7e8121..a09f822 100644 Binary files a/_module/ncs/quest_cloak6.ncs and b/_module/ncs/quest_cloak6.ncs differ diff --git a/_module/ncs/quest_cloak7.ncs b/_module/ncs/quest_cloak7.ncs index 054bcc4..8ab003a 100644 Binary files a/_module/ncs/quest_cloak7.ncs and b/_module/ncs/quest_cloak7.ncs differ diff --git a/_module/ncs/quest_dic1.ncs b/_module/ncs/quest_dic1.ncs index 370c317..4e6e773 100644 Binary files a/_module/ncs/quest_dic1.ncs and b/_module/ncs/quest_dic1.ncs differ diff --git a/_module/ncs/quest_dic2.ncs b/_module/ncs/quest_dic2.ncs index 64c4a01..bf0f76d 100644 Binary files a/_module/ncs/quest_dic2.ncs and b/_module/ncs/quest_dic2.ncs differ diff --git a/_module/ncs/quest_dic3.ncs b/_module/ncs/quest_dic3.ncs index 903323e..1211dff 100644 Binary files a/_module/ncs/quest_dic3.ncs and b/_module/ncs/quest_dic3.ncs differ diff --git a/_module/ncs/quest_elf1.ncs b/_module/ncs/quest_elf1.ncs index b1b8a54..1ce8d21 100644 Binary files a/_module/ncs/quest_elf1.ncs and b/_module/ncs/quest_elf1.ncs differ diff --git a/_module/ncs/quest_elv1.ncs b/_module/ncs/quest_elv1.ncs index b1b8a54..1ce8d21 100644 Binary files a/_module/ncs/quest_elv1.ncs and b/_module/ncs/quest_elv1.ncs differ diff --git a/_module/ncs/quest_elv2.ncs b/_module/ncs/quest_elv2.ncs index fc07692..85e9136 100644 Binary files a/_module/ncs/quest_elv2.ncs and b/_module/ncs/quest_elv2.ncs differ diff --git a/_module/ncs/quest_elv3.ncs b/_module/ncs/quest_elv3.ncs index 8f56bc0..d4fce4b 100644 Binary files a/_module/ncs/quest_elv3.ncs and b/_module/ncs/quest_elv3.ncs differ diff --git a/_module/ncs/quest_elv4.ncs b/_module/ncs/quest_elv4.ncs index ff171ac..ed7f59a 100644 Binary files a/_module/ncs/quest_elv4.ncs and b/_module/ncs/quest_elv4.ncs differ diff --git a/_module/ncs/quest_elv5.ncs b/_module/ncs/quest_elv5.ncs index 61b3d98..d655e86 100644 Binary files a/_module/ncs/quest_elv5.ncs and b/_module/ncs/quest_elv5.ncs differ diff --git a/_module/ncs/quest_elv6.ncs b/_module/ncs/quest_elv6.ncs index 6855c23..8e821c3 100644 Binary files a/_module/ncs/quest_elv6.ncs and b/_module/ncs/quest_elv6.ncs differ diff --git a/_module/ncs/quest_elv8.ncs b/_module/ncs/quest_elv8.ncs index ff171ac..ed7f59a 100644 Binary files a/_module/ncs/quest_elv8.ncs and b/_module/ncs/quest_elv8.ncs differ diff --git a/_module/ncs/quest_elv9.ncs b/_module/ncs/quest_elv9.ncs index 89b378d..0b8b021 100644 Binary files a/_module/ncs/quest_elv9.ncs and b/_module/ncs/quest_elv9.ncs differ diff --git a/_module/ncs/quest_jerkin1.ncs b/_module/ncs/quest_jerkin1.ncs index c7d41d1..ba7f29f 100644 Binary files a/_module/ncs/quest_jerkin1.ncs and b/_module/ncs/quest_jerkin1.ncs differ diff --git a/_module/ncs/quest_jerkin3.ncs b/_module/ncs/quest_jerkin3.ncs index 8ab5483..6bcb36d 100644 Binary files a/_module/ncs/quest_jerkin3.ncs and b/_module/ncs/quest_jerkin3.ncs differ diff --git a/_module/ncs/quest_jerkin4.ncs b/_module/ncs/quest_jerkin4.ncs index 1791740..20592dd 100644 Binary files a/_module/ncs/quest_jerkin4.ncs and b/_module/ncs/quest_jerkin4.ncs differ diff --git a/_module/ncs/quest_jerkindone.ncs b/_module/ncs/quest_jerkindone.ncs index 3e48122..ca5e7a3 100644 Binary files a/_module/ncs/quest_jerkindone.ncs and b/_module/ncs/quest_jerkindone.ncs differ diff --git a/_module/ncs/quest_jerkins5.ncs b/_module/ncs/quest_jerkins5.ncs index 1308fb9..77e6c87 100644 Binary files a/_module/ncs/quest_jerkins5.ncs and b/_module/ncs/quest_jerkins5.ncs differ diff --git a/_module/ncs/quest_jim1.ncs b/_module/ncs/quest_jim1.ncs index b4f1002..bd7c6c1 100644 Binary files a/_module/ncs/quest_jim1.ncs and b/_module/ncs/quest_jim1.ncs differ diff --git a/_module/ncs/quest_jim2.ncs b/_module/ncs/quest_jim2.ncs index 6264d09..385c566 100644 Binary files a/_module/ncs/quest_jim2.ncs and b/_module/ncs/quest_jim2.ncs differ diff --git a/_module/ncs/quest_jim5.ncs b/_module/ncs/quest_jim5.ncs index b5e6854..67d6184 100644 Binary files a/_module/ncs/quest_jim5.ncs and b/_module/ncs/quest_jim5.ncs differ diff --git a/_module/ncs/quest_jim6.ncs b/_module/ncs/quest_jim6.ncs index 28e3f2e..885216e 100644 Binary files a/_module/ncs/quest_jim6.ncs and b/_module/ncs/quest_jim6.ncs differ diff --git a/_module/ncs/quest_jimmy8.ncs b/_module/ncs/quest_jimmy8.ncs index aa6ec4b..726c249 100644 Binary files a/_module/ncs/quest_jimmy8.ncs and b/_module/ncs/quest_jimmy8.ncs differ diff --git a/_module/ncs/quest_jimmy9.ncs b/_module/ncs/quest_jimmy9.ncs index a43aba3..2465752 100644 Binary files a/_module/ncs/quest_jimmy9.ncs and b/_module/ncs/quest_jimmy9.ncs differ diff --git a/_module/ncs/quest_lib2.ncs b/_module/ncs/quest_lib2.ncs index 4c2d7d8..3b4dda6 100644 Binary files a/_module/ncs/quest_lib2.ncs and b/_module/ncs/quest_lib2.ncs differ diff --git a/_module/ncs/quest_library.ncs b/_module/ncs/quest_library.ncs index e72ebe4..5346f75 100644 Binary files a/_module/ncs/quest_library.ncs and b/_module/ncs/quest_library.ncs differ diff --git a/_module/ncs/quest_library2.ncs b/_module/ncs/quest_library2.ncs index a0e3d70..b580c4d 100644 Binary files a/_module/ncs/quest_library2.ncs and b/_module/ncs/quest_library2.ncs differ diff --git a/_module/ncs/quest_library3.ncs b/_module/ncs/quest_library3.ncs index 4d541be..6a9e7f2 100644 Binary files a/_module/ncs/quest_library3.ncs and b/_module/ncs/quest_library3.ncs differ diff --git a/_module/ncs/quest_lith1.ncs b/_module/ncs/quest_lith1.ncs index b576b04..58899a3 100644 Binary files a/_module/ncs/quest_lith1.ncs and b/_module/ncs/quest_lith1.ncs differ diff --git a/_module/ncs/quest_lith2.ncs b/_module/ncs/quest_lith2.ncs index 65095fd..54ba54b 100644 Binary files a/_module/ncs/quest_lith2.ncs and b/_module/ncs/quest_lith2.ncs differ diff --git a/_module/ncs/quest_mayor1.ncs b/_module/ncs/quest_mayor1.ncs index 363e016..b0426af 100644 Binary files a/_module/ncs/quest_mayor1.ncs and b/_module/ncs/quest_mayor1.ncs differ diff --git a/_module/ncs/quest_mayor2.ncs b/_module/ncs/quest_mayor2.ncs index 5eaad64..8afa977 100644 Binary files a/_module/ncs/quest_mayor2.ncs and b/_module/ncs/quest_mayor2.ncs differ diff --git a/_module/ncs/quest_mayor3.ncs b/_module/ncs/quest_mayor3.ncs index 57827b7..5501e96 100644 Binary files a/_module/ncs/quest_mayor3.ncs and b/_module/ncs/quest_mayor3.ncs differ diff --git a/_module/ncs/quest_night1.ncs b/_module/ncs/quest_night1.ncs index 495b295..1ee97ba 100644 Binary files a/_module/ncs/quest_night1.ncs and b/_module/ncs/quest_night1.ncs differ diff --git a/_module/ncs/quest_night2.ncs b/_module/ncs/quest_night2.ncs index d38c20a..2b179fb 100644 Binary files a/_module/ncs/quest_night2.ncs and b/_module/ncs/quest_night2.ncs differ diff --git a/_module/ncs/quest_repent1.ncs b/_module/ncs/quest_repent1.ncs index 9ae108f..bd3c274 100644 Binary files a/_module/ncs/quest_repent1.ncs and b/_module/ncs/quest_repent1.ncs differ diff --git a/_module/ncs/quest_repent2.ncs b/_module/ncs/quest_repent2.ncs index fea6ad3..c29efa8 100644 Binary files a/_module/ncs/quest_repent2.ncs and b/_module/ncs/quest_repent2.ncs differ diff --git a/_module/ncs/quest_repent3.ncs b/_module/ncs/quest_repent3.ncs index 02ff61a..58d8aa8 100644 Binary files a/_module/ncs/quest_repent3.ncs and b/_module/ncs/quest_repent3.ncs differ diff --git a/_module/ncs/quest_tm1.ncs b/_module/ncs/quest_tm1.ncs index 4333b8d..f03b68d 100644 Binary files a/_module/ncs/quest_tm1.ncs and b/_module/ncs/quest_tm1.ncs differ diff --git a/_module/ncs/quest_tm2.ncs b/_module/ncs/quest_tm2.ncs index 939e81e..8a406d2 100644 Binary files a/_module/ncs/quest_tm2.ncs and b/_module/ncs/quest_tm2.ncs differ diff --git a/_module/ncs/quest_tm3.ncs b/_module/ncs/quest_tm3.ncs index b4814d4..97b1fa3 100644 Binary files a/_module/ncs/quest_tm3.ncs and b/_module/ncs/quest_tm3.ncs differ diff --git a/_module/ncs/quest_trin1.ncs b/_module/ncs/quest_trin1.ncs index c261113..9e9e3fe 100644 Binary files a/_module/ncs/quest_trin1.ncs and b/_module/ncs/quest_trin1.ncs differ diff --git a/_module/ncs/quest_trin2.ncs b/_module/ncs/quest_trin2.ncs index 17d607d..fa75ee3 100644 Binary files a/_module/ncs/quest_trin2.ncs and b/_module/ncs/quest_trin2.ncs differ diff --git a/_module/ncs/quest_unen1.ncs b/_module/ncs/quest_unen1.ncs index 0789ae9..33e3cca 100644 Binary files a/_module/ncs/quest_unen1.ncs and b/_module/ncs/quest_unen1.ncs differ diff --git a/_module/ncs/quest_unen2.ncs b/_module/ncs/quest_unen2.ncs index fc95603..c18d46f 100644 Binary files a/_module/ncs/quest_unen2.ncs and b/_module/ncs/quest_unen2.ncs differ diff --git a/_module/ncs/quest_unen3.ncs b/_module/ncs/quest_unen3.ncs index 429f9c2..7e50c05 100644 Binary files a/_module/ncs/quest_unen3.ncs and b/_module/ncs/quest_unen3.ncs differ diff --git a/_module/ncs/ra_close_door.ncs b/_module/ncs/ra_close_door.ncs new file mode 100644 index 0000000..2876a3e Binary files /dev/null and b/_module/ncs/ra_close_door.ncs differ diff --git a/_module/ncs/res_heartbeat.ncs b/_module/ncs/res_heartbeat.ncs index 2508bdb..f77b4e7 100644 Binary files a/_module/ncs/res_heartbeat.ncs and b/_module/ncs/res_heartbeat.ncs differ diff --git a/_module/ncs/runwaypoints.ncs b/_module/ncs/runwaypoints.ncs index e5e49ae..c9901bd 100644 Binary files a/_module/ncs/runwaypoints.ncs and b/_module/ncs/runwaypoints.ncs differ diff --git a/_module/ncs/sc_headless01.ncs b/_module/ncs/sc_headless01.ncs index 2243025..dfe8aab 100644 Binary files a/_module/ncs/sc_headless01.ncs and b/_module/ncs/sc_headless01.ncs differ diff --git a/_module/ncs/sc_headless02.ncs b/_module/ncs/sc_headless02.ncs index 50a1deb..084dc86 100644 Binary files a/_module/ncs/sc_headless02.ncs and b/_module/ncs/sc_headless02.ncs differ diff --git a/_module/ncs/sc_mared05.ncs b/_module/ncs/sc_mared05.ncs index fd6c563..9c8a113 100644 Binary files a/_module/ncs/sc_mared05.ncs and b/_module/ncs/sc_mared05.ncs differ diff --git a/_module/ncs/se_door_death.ncs b/_module/ncs/se_door_death.ncs new file mode 100644 index 0000000..4ff9fd1 Binary files /dev/null and b/_module/ncs/se_door_death.ncs differ diff --git a/_module/ncs/sei_sit.ncs b/_module/ncs/sei_sit.ncs new file mode 100644 index 0000000..ba8cb20 Binary files /dev/null and b/_module/ncs/sei_sit.ncs differ diff --git a/_module/ncs/shogun_portal.ncs b/_module/ncs/shogun_portal.ncs index d0700bd..cb0d6a5 100644 Binary files a/_module/ncs/shogun_portal.ncs and b/_module/ncs/shogun_portal.ncs differ diff --git a/_module/ncs/sign_level_check.ncs b/_module/ncs/sign_level_check.ncs index 4195151..aa84348 100644 Binary files a/_module/ncs/sign_level_check.ncs and b/_module/ncs/sign_level_check.ncs differ diff --git a/_module/ncs/sob_examine002.ncs b/_module/ncs/sob_examine002.ncs new file mode 100644 index 0000000..26d4486 Binary files /dev/null and b/_module/ncs/sob_examine002.ncs differ diff --git a/_module/ncs/spirit_item_chk.ncs b/_module/ncs/spirit_item_chk.ncs index 75d9e94..1731cfd 100644 Binary files a/_module/ncs/spirit_item_chk.ncs and b/_module/ncs/spirit_item_chk.ncs differ diff --git a/_module/ncs/spirit_item_take.ncs b/_module/ncs/spirit_item_take.ncs index 358eb18..9052ac3 100644 Binary files a/_module/ncs/spirit_item_take.ncs and b/_module/ncs/spirit_item_take.ncs differ diff --git a/_module/ncs/spirit_lev_chk.ncs b/_module/ncs/spirit_lev_chk.ncs index 80e015c..48ed2af 100644 Binary files a/_module/ncs/spirit_lev_chk.ncs and b/_module/ncs/spirit_lev_chk.ncs differ diff --git a/_module/ncs/spirit_monk_chk.ncs b/_module/ncs/spirit_monk_chk.ncs index fbb4e2b..556e73e 100644 Binary files a/_module/ncs/spirit_monk_chk.ncs and b/_module/ncs/spirit_monk_chk.ncs differ diff --git a/_module/ncs/spirit_rewrd_chk.ncs b/_module/ncs/spirit_rewrd_chk.ncs index 57e9fb7..2c85d58 100644 Binary files a/_module/ncs/spirit_rewrd_chk.ncs and b/_module/ncs/spirit_rewrd_chk.ncs differ diff --git a/_module/ncs/sr_oce_hdragon.ncs b/_module/ncs/sr_oce_hdragon.ncs index 7faf634..2d4b1d6 100644 Binary files a/_module/ncs/sr_oce_hdragon.ncs and b/_module/ncs/sr_oce_hdragon.ncs differ diff --git a/_module/ncs/statue_conv.ncs b/_module/ncs/statue_conv.ncs index e7d9e61..8243139 100644 Binary files a/_module/ncs/statue_conv.ncs and b/_module/ncs/statue_conv.ncs differ diff --git a/_module/ncs/stoneofreturn.ncs b/_module/ncs/stoneofreturn.ncs index 1a6d428..d0ff771 100644 Binary files a/_module/ncs/stoneofreturn.ncs and b/_module/ncs/stoneofreturn.ncs differ diff --git a/_module/ncs/strring_gen.ncs b/_module/ncs/strring_gen.ncs index ab33ef1..3814af3 100644 Binary files a/_module/ncs/strring_gen.ncs and b/_module/ncs/strring_gen.ncs differ diff --git a/_module/ncs/tab_activateitem.ncs b/_module/ncs/tab_activateitem.ncs index 139f900..725a1ea 100644 Binary files a/_module/ncs/tab_activateitem.ncs and b/_module/ncs/tab_activateitem.ncs differ diff --git a/_module/ncs/tab_boulder1.ncs b/_module/ncs/tab_boulder1.ncs index b8440bc..b30b067 100644 Binary files a/_module/ncs/tab_boulder1.ncs and b/_module/ncs/tab_boulder1.ncs differ diff --git a/_module/ncs/tab_chest.ncs b/_module/ncs/tab_chest.ncs index c33f2d0..1618553 100644 Binary files a/_module/ncs/tab_chest.ncs and b/_module/ncs/tab_chest.ncs differ diff --git a/_module/ncs/tab_coldscript.ncs b/_module/ncs/tab_coldscript.ncs index 24886c5..d70ab9f 100644 Binary files a/_module/ncs/tab_coldscript.ncs and b/_module/ncs/tab_coldscript.ncs differ diff --git a/_module/ncs/tab_containregen.ncs b/_module/ncs/tab_containregen.ncs index b8440bc..b30b067 100644 Binary files a/_module/ncs/tab_containregen.ncs and b/_module/ncs/tab_containregen.ncs differ diff --git a/_module/ncs/tab_glow.ncs b/_module/ncs/tab_glow.ncs index 9a14219..6691481 100644 Binary files a/_module/ncs/tab_glow.ncs and b/_module/ncs/tab_glow.ncs differ diff --git a/_module/ncs/tab_glow_blue.ncs b/_module/ncs/tab_glow_blue.ncs index 9a14219..6691481 100644 Binary files a/_module/ncs/tab_glow_blue.ncs and b/_module/ncs/tab_glow_blue.ncs differ diff --git a/_module/ncs/tab_glow_red.ncs b/_module/ncs/tab_glow_red.ncs index 6f0c888..b194db6 100644 Binary files a/_module/ncs/tab_glow_red.ncs and b/_module/ncs/tab_glow_red.ncs differ diff --git a/_module/ncs/tab_heartbeat.ncs b/_module/ncs/tab_heartbeat.ncs index 0206111..f1834f5 100644 Binary files a/_module/ncs/tab_heartbeat.ncs and b/_module/ncs/tab_heartbeat.ncs differ diff --git a/_module/ncs/tab_newply_port.ncs b/_module/ncs/tab_newply_port.ncs index 7d0909f..493c38e 100644 Binary files a/_module/ncs/tab_newply_port.ncs and b/_module/ncs/tab_newply_port.ncs differ diff --git a/_module/ncs/tab_newply_port2.ncs b/_module/ncs/tab_newply_port2.ncs index 35edfa4..b8c3f0f 100644 Binary files a/_module/ncs/tab_newply_port2.ncs and b/_module/ncs/tab_newply_port2.ncs differ diff --git a/_module/ncs/tab_onarea_enter.ncs b/_module/ncs/tab_onarea_enter.ncs index 1895886..42980cf 100644 Binary files a/_module/ncs/tab_onarea_enter.ncs and b/_module/ncs/tab_onarea_enter.ncs differ diff --git a/_module/ncs/tab_onenter.ncs b/_module/ncs/tab_onenter.ncs index 4b58e4a..cd9aa17 100644 Binary files a/_module/ncs/tab_onenter.ncs and b/_module/ncs/tab_onenter.ncs differ diff --git a/_module/ncs/tab_onlevelup.ncs b/_module/ncs/tab_onlevelup.ncs index d75d2f6..9d43d77 100644 Binary files a/_module/ncs/tab_onlevelup.ncs and b/_module/ncs/tab_onlevelup.ncs differ diff --git a/_module/ncs/tab_onrest.ncs b/_module/ncs/tab_onrest.ncs index 170d27e..fcd8a35 100644 Binary files a/_module/ncs/tab_onrest.ncs and b/_module/ncs/tab_onrest.ncs differ diff --git a/_module/ncs/tab_respawn.ncs b/_module/ncs/tab_respawn.ncs index 32b702f..4fa2526 100644 Binary files a/_module/ncs/tab_respawn.ncs and b/_module/ncs/tab_respawn.ncs differ diff --git a/_module/ncs/tab_spawn_animat.ncs b/_module/ncs/tab_spawn_animat.ncs index b470f15..7a84836 100644 Binary files a/_module/ncs/tab_spawn_animat.ncs and b/_module/ncs/tab_spawn_animat.ncs differ diff --git a/_module/ncs/tab_stat_trig.ncs b/_module/ncs/tab_stat_trig.ncs index 847022f..0aab363 100644 Binary files a/_module/ncs/tab_stat_trig.ncs and b/_module/ncs/tab_stat_trig.ncs differ diff --git a/_module/ncs/tab_xp_multi.ncs b/_module/ncs/tab_xp_multi.ncs index 7116899..1a2600a 100644 Binary files a/_module/ncs/tab_xp_multi.ncs and b/_module/ncs/tab_xp_multi.ncs differ diff --git a/_module/ncs/tab_xpscript.ncs b/_module/ncs/tab_xpscript.ncs index 50ac0dd..189eb8c 100644 Binary files a/_module/ncs/tab_xpscript.ncs and b/_module/ncs/tab_xpscript.ncs differ diff --git a/_module/ncs/tab_xpscript2.ncs b/_module/ncs/tab_xpscript2.ncs index 8eb0b5b..30b9a54 100644 Binary files a/_module/ncs/tab_xpscript2.ncs and b/_module/ncs/tab_xpscript2.ncs differ diff --git a/_module/ncs/tab_xpscript3.ncs b/_module/ncs/tab_xpscript3.ncs index 3b53d8f..5567928 100644 Binary files a/_module/ncs/tab_xpscript3.ncs and b/_module/ncs/tab_xpscript3.ncs differ diff --git a/_module/ncs/tab_xpscript4.ncs b/_module/ncs/tab_xpscript4.ncs index f74c344..790c20c 100644 Binary files a/_module/ncs/tab_xpscript4.ncs and b/_module/ncs/tab_xpscript4.ncs differ diff --git a/_module/ncs/tm_chest.ncs b/_module/ncs/tm_chest.ncs index 208d172..dabffad 100644 Binary files a/_module/ncs/tm_chest.ncs and b/_module/ncs/tm_chest.ncs differ diff --git a/_module/ncs/trap_dis_1k.ncs b/_module/ncs/trap_dis_1k.ncs index 93aa2e4..4ddc41b 100644 Binary files a/_module/ncs/trap_dis_1k.ncs and b/_module/ncs/trap_dis_1k.ncs differ diff --git a/_module/ncs/trap_dis_2k.ncs b/_module/ncs/trap_dis_2k.ncs index 49e3790..4f97177 100644 Binary files a/_module/ncs/trap_dis_2k.ncs and b/_module/ncs/trap_dis_2k.ncs differ diff --git a/_module/ncs/trap_dis_5k.ncs b/_module/ncs/trap_dis_5k.ncs index 07bc1c5..e90f7fd 100644 Binary files a/_module/ncs/trap_dis_5k.ncs and b/_module/ncs/trap_dis_5k.ncs differ diff --git a/_module/ncs/trinity_activate.ncs b/_module/ncs/trinity_activate.ncs index 4555fc9..351087f 100644 Binary files a/_module/ncs/trinity_activate.ncs and b/_module/ncs/trinity_activate.ncs differ diff --git a/_module/ncs/ww_ch_ac9.ncs b/_module/ncs/ww_ch_ac9.ncs index f952de4..9f63c72 100644 Binary files a/_module/ncs/ww_ch_ac9.ncs and b/_module/ncs/ww_ch_ac9.ncs differ diff --git a/_module/ncs/x2_im_cancel.ncs b/_module/ncs/x2_im_cancel.ncs index 50e48ba..b900c36 100644 Binary files a/_module/ncs/x2_im_cancel.ncs and b/_module/ncs/x2_im_cancel.ncs differ diff --git a/_module/ncs/x2_tab_stealth.ncs b/_module/ncs/x2_tab_stealth.ncs index 76673d5..eb211ad 100644 Binary files a/_module/ncs/x2_tab_stealth.ncs and b/_module/ncs/x2_tab_stealth.ncs differ diff --git a/_module/ncs/x_foreged_short.ncs b/_module/ncs/x_foreged_short.ncs index 04c3060..bbc13cc 100644 Binary files a/_module/ncs/x_foreged_short.ncs and b/_module/ncs/x_foreged_short.ncs differ diff --git a/_module/ncs/x_forge_armor.ncs b/_module/ncs/x_forge_armor.ncs index 0f7eb05..b7008dd 100644 Binary files a/_module/ncs/x_forge_armor.ncs and b/_module/ncs/x_forge_armor.ncs differ diff --git a/_module/ncs/x_forge_boots.ncs b/_module/ncs/x_forge_boots.ncs index b352764..161a4c7 100644 Binary files a/_module/ncs/x_forge_boots.ncs and b/_module/ncs/x_forge_boots.ncs differ diff --git a/_module/ncs/x_forge_cloak.ncs b/_module/ncs/x_forge_cloak.ncs index 685091f..a0f8562 100644 Binary files a/_module/ncs/x_forge_cloak.ncs and b/_module/ncs/x_forge_cloak.ncs differ diff --git a/_module/ncs/x_forge_helmet.ncs b/_module/ncs/x_forge_helmet.ncs index 4d56b43..8dcf4be 100644 Binary files a/_module/ncs/x_forge_helmet.ncs and b/_module/ncs/x_forge_helmet.ncs differ diff --git a/_module/ncs/x_forge_long.ncs b/_module/ncs/x_forge_long.ncs index a02a77e..5ca7f95 100644 Binary files a/_module/ncs/x_forge_long.ncs and b/_module/ncs/x_forge_long.ncs differ diff --git a/_module/ncs/x_forge_morkain.ncs b/_module/ncs/x_forge_morkain.ncs index 2c3d959..68a0560 100644 Binary files a/_module/ncs/x_forge_morkain.ncs and b/_module/ncs/x_forge_morkain.ncs differ diff --git a/_module/ncs/x_forge_shield.ncs b/_module/ncs/x_forge_shield.ncs index d115c6b..4f884d0 100644 Binary files a/_module/ncs/x_forge_shield.ncs and b/_module/ncs/x_forge_shield.ncs differ diff --git a/_module/ncs/xx.ncs b/_module/ncs/xx.ncs index 49e72e0..0898983 100644 Binary files a/_module/ncs/xx.ncs and b/_module/ncs/xx.ncs differ diff --git a/_module/ncs/xx_atlport.ncs b/_module/ncs/xx_atlport.ncs index b41b7aa..c33b4ee 100644 Binary files a/_module/ncs/xx_atlport.ncs and b/_module/ncs/xx_atlport.ncs differ diff --git a/_module/ncs/xx_cb_bleed.ncs b/_module/ncs/xx_cb_bleed.ncs index b3cba48..d77cd8d 100644 Binary files a/_module/ncs/xx_cb_bleed.ncs and b/_module/ncs/xx_cb_bleed.ncs differ diff --git a/_module/ncs/xx_cb_death.ncs b/_module/ncs/xx_cb_death.ncs index cbb272d..e965083 100644 Binary files a/_module/ncs/xx_cb_death.ncs and b/_module/ncs/xx_cb_death.ncs differ diff --git a/_module/ncs/xx_cb_enter.ncs b/_module/ncs/xx_cb_enter.ncs index aeeb711..f51b187 100644 Binary files a/_module/ncs/xx_cb_enter.ncs and b/_module/ncs/xx_cb_enter.ncs differ diff --git a/_module/ncs/xx_cb_exit.ncs b/_module/ncs/xx_cb_exit.ncs index ed76cc4..3466b00 100644 Binary files a/_module/ncs/xx_cb_exit.ncs and b/_module/ncs/xx_cb_exit.ncs differ diff --git a/_module/ncs/xx_cb_hb.ncs b/_module/ncs/xx_cb_hb.ncs index 9eff572..a5197b5 100644 Binary files a/_module/ncs/xx_cb_hb.ncs and b/_module/ncs/xx_cb_hb.ncs differ diff --git a/_module/ncs/xx_cb_pvp_respaw.ncs b/_module/ncs/xx_cb_pvp_respaw.ncs index ee8df07..57a26f9 100644 Binary files a/_module/ncs/xx_cb_pvp_respaw.ncs and b/_module/ncs/xx_cb_pvp_respaw.ncs differ diff --git a/_module/ncs/xx_cb_respawn.ncs b/_module/ncs/xx_cb_respawn.ncs index bc24a71..b01fe66 100644 Binary files a/_module/ncs/xx_cb_respawn.ncs and b/_module/ncs/xx_cb_respawn.ncs differ diff --git a/_module/ncs/xx_chk_lvl15.ncs b/_module/ncs/xx_chk_lvl15.ncs index 1c370ab..309ba87 100644 Binary files a/_module/ncs/xx_chk_lvl15.ncs and b/_module/ncs/xx_chk_lvl15.ncs differ diff --git a/_module/ncs/xx_dragonport.ncs b/_module/ncs/xx_dragonport.ncs index 66d62ca..15eee1c 100644 Binary files a/_module/ncs/xx_dragonport.ncs and b/_module/ncs/xx_dragonport.ncs differ diff --git a/_module/ncs/xx_dwarfport.ncs b/_module/ncs/xx_dwarfport.ncs index 76638f5..6a4519c 100644 Binary files a/_module/ncs/xx_dwarfport.ncs and b/_module/ncs/xx_dwarfport.ncs differ diff --git a/_module/ncs/xx_manyport.ncs b/_module/ncs/xx_manyport.ncs index 3b00563..b35f37b 100644 Binary files a/_module/ncs/xx_manyport.ncs and b/_module/ncs/xx_manyport.ncs differ diff --git a/_module/ncs/xx_nordicport.ncs b/_module/ncs/xx_nordicport.ncs index 2334e04..f11fa33 100644 Binary files a/_module/ncs/xx_nordicport.ncs and b/_module/ncs/xx_nordicport.ncs differ diff --git a/_module/ncs/xx_ques_add.ncs b/_module/ncs/xx_ques_add.ncs index 43a7eec..85d0f2a 100644 Binary files a/_module/ncs/xx_ques_add.ncs and b/_module/ncs/xx_ques_add.ncs differ diff --git a/_module/ncs/xx_ques_chest.ncs b/_module/ncs/xx_ques_chest.ncs index f055491..ade6f1d 100644 Binary files a/_module/ncs/xx_ques_chest.ncs and b/_module/ncs/xx_ques_chest.ncs differ diff --git a/_module/ncs/xx_ques_finished.ncs b/_module/ncs/xx_ques_finished.ncs index d25ba73..da7663e 100644 Binary files a/_module/ncs/xx_ques_finished.ncs and b/_module/ncs/xx_ques_finished.ncs differ diff --git a/_module/ncs/xx_ques_progress.ncs b/_module/ncs/xx_ques_progress.ncs index ce11568..d9c9f2a 100644 Binary files a/_module/ncs/xx_ques_progress.ncs and b/_module/ncs/xx_ques_progress.ncs differ diff --git a/_module/ncs/xx_ques_retrieve.ncs b/_module/ncs/xx_ques_retrieve.ncs index 639b554..33a44ec 100644 Binary files a/_module/ncs/xx_ques_retrieve.ncs and b/_module/ncs/xx_ques_retrieve.ncs differ diff --git a/_module/ncs/xx_quest_start.ncs b/_module/ncs/xx_quest_start.ncs index 1b9ce34..004cc85 100644 Binary files a/_module/ncs/xx_quest_start.ncs and b/_module/ncs/xx_quest_start.ncs differ diff --git a/_module/ncs/xx_remove_quest.ncs b/_module/ncs/xx_remove_quest.ncs index 7d1d531..d0e7a41 100644 Binary files a/_module/ncs/xx_remove_quest.ncs and b/_module/ncs/xx_remove_quest.ncs differ diff --git a/_module/ncs/xx_reward10k.ncs b/_module/ncs/xx_reward10k.ncs index 34afac6..882f912 100644 Binary files a/_module/ncs/xx_reward10k.ncs and b/_module/ncs/xx_reward10k.ncs differ diff --git a/_module/ncs/xx_reward2k.ncs b/_module/ncs/xx_reward2k.ncs index efa6bc1..d975fc0 100644 Binary files a/_module/ncs/xx_reward2k.ncs and b/_module/ncs/xx_reward2k.ncs differ diff --git a/_module/ncs/xx_tempcain.ncs b/_module/ncs/xx_tempcain.ncs index 790de34..bed5970 100644 Binary files a/_module/ncs/xx_tempcain.ncs and b/_module/ncs/xx_tempcain.ncs differ diff --git a/_module/ncs/xx_testreward.ncs b/_module/ncs/xx_testreward.ncs index 4b3bf6e..0d655d8 100644 Binary files a/_module/ncs/xx_testreward.ncs and b/_module/ncs/xx_testreward.ncs differ diff --git a/_module/ncs/xx_trinport.ncs b/_module/ncs/xx_trinport.ncs index 970f647..68800c2 100644 Binary files a/_module/ncs/xx_trinport.ncs and b/_module/ncs/xx_trinport.ncs differ diff --git a/_module/ncs/zach_stat_trig.ncs b/_module/ncs/zach_stat_trig.ncs index d2194f6..195492a 100644 Binary files a/_module/ncs/zach_stat_trig.ncs and b/_module/ncs/zach_stat_trig.ncs differ diff --git a/_module/ncs/zedd_check.ncs b/_module/ncs/zedd_check.ncs index 00e0b4d..d556cbe 100644 Binary files a/_module/ncs/zedd_check.ncs and b/_module/ncs/zedd_check.ncs differ diff --git a/_module/ncs/zedd_mc_port.ncs b/_module/ncs/zedd_mc_port.ncs index a043837..699706f 100644 Binary files a/_module/ncs/zedd_mc_port.ncs and b/_module/ncs/zedd_mc_port.ncs differ diff --git a/_module/nss/at_ent_dbi_room.nss b/_module/nss/at_ent_dbi_room.nss new file mode 100644 index 0000000..bc1e3b7 --- /dev/null +++ b/_module/nss/at_ent_dbi_room.nss @@ -0,0 +1,44 @@ +void main() +{ +//:: Declare major variables + object oPC = GetPCSpeaker(); + object oBaseArea = GetObjectByTag("DBI_PC_ROOMS"); + + string sPCName = GetName(oPC); + string sBaseResRef = GetResRef(oBaseArea); + string sBaseName = GetName(oBaseArea); + string sNewTag = sPCName+sBaseResRef; + string sNewName = sPCName+"'s Room at the Dragonback Inn"; + +//:: Create instanced area from player's information + CreateArea(sBaseResRef, sNewTag, sNewName); + +//:: Get information about new area instance + object oNewRoom = GetObjectByTag(sNewTag); + string sNewResRef = GetResRef(oNewRoom); + + + location lEntry = GetLocation(GetObjectByTag("DBI_ROOM_MAT")); + + +//:: Create entry waypoint in new area instance + CreateObject(OBJECT_TYPE_WAYPOINT, "nw_waypoint001", lEntry, FALSE, "WP_"+sNewTag); + +//:: Make sure the area is valid + object oTarget; + location lTarget; + oTarget = GetWaypointByTag("WP_"+sNewTag); + + lTarget = GetLocation(oTarget); + + if (GetAreaFromLocation(lTarget)==OBJECT_INVALID) + { + SendMessageToPC(oPC, "Warning: Area does not exist!"); + return; + } +//:: Move PC to instanced area. + AssignCommand(oPC, ClearAllActions()); + + AssignCommand(oPC, ActionJumpToLocation(lTarget)); + +} diff --git a/_module/nss/at_enter_pc_room.nss b/_module/nss/at_enter_pc_room.nss new file mode 100644 index 0000000..42bd1c7 --- /dev/null +++ b/_module/nss/at_enter_pc_room.nss @@ -0,0 +1,44 @@ +void main() +{ +//:: Declare major variables + object oPC = GetPCSpeaker(); + object oBaseArea = GetObjectByTag("IFM_PC_ROOMS"); + + string sPCName = GetName(oPC); + string sBaseResRef = GetResRef(oBaseArea); + string sBaseName = GetName(oBaseArea); + string sNewTag = sPCName+sBaseResRef; + string sNewName = sPCName+"'s Room at the Flying Monkey"; + +//:: Create instanced area from player's information + CreateArea(sBaseResRef, sNewTag, sNewName); + +//:: Get information about new area instance + object oNewRoom = GetObjectByTag(sNewTag); + string sNewResRef = GetResRef(oNewRoom); + + + location lEntry = GetLocation(GetObjectByTag("PC_ROOM_MAT")); + + +//:: Create entry waypoint in new area instance + CreateObject(OBJECT_TYPE_WAYPOINT, "nw_waypoint001", lEntry, FALSE, "WP_"+sNewTag); + +//:: Make sure the area is valid + object oTarget; + location lTarget; + oTarget = GetWaypointByTag("WP_"+sNewTag); + + lTarget = GetLocation(oTarget); + + if (GetAreaFromLocation(lTarget)==OBJECT_INVALID) + { + SendMessageToPC(oPC, "Warning: Area does not exist!"); + return; + } +//:: Move PC to instanced area. + AssignCommand(oPC, ClearAllActions()); + + AssignCommand(oPC, ActionJumpToLocation(lTarget)); + +} \ No newline at end of file diff --git a/_module/nss/cv_inn_door.nss b/_module/nss/cv_inn_door.nss new file mode 100644 index 0000000..a9bb023 --- /dev/null +++ b/_module/nss/cv_inn_door.nss @@ -0,0 +1,10 @@ +void main() +{ + +object oPC = GetClickingObject(); + +if (!GetIsPC(oPC)) return; + +ActionStartConversation(oPC, ""); + +} diff --git a/_module/nss/hif_onmoduleload.nss b/_module/nss/hif_onmoduleload.nss index 78681c0..940243e 100644 --- a/_module/nss/hif_onmoduleload.nss +++ b/_module/nss/hif_onmoduleload.nss @@ -5,9 +5,16 @@ // multiple handlers for the onmoduleload event. // ///////////////////////////////////////////////////////////////////// +#include "nui_i_main" void main() { - ExecuteScript("prc_onmodload", OBJECT_SELF); + SetEventScript(GetModule(), EVENT_SCRIPT_MODULE_ON_NUI_EVENT, "mod_events"); + SetEventScript(GetModule(), EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET, "mod_events"); + + NUI_Initialize(); + NUI_HandleEvents(); + + ExecuteScript("prc_onmodload", OBJECT_SELF); ExecuteScript("trin_modload", OBJECT_SELF); } diff --git a/_module/nss/mod_events.nss b/_module/nss/mod_events.nss new file mode 100644 index 0000000..85d3756 --- /dev/null +++ b/_module/nss/mod_events.nss @@ -0,0 +1,23 @@ +#include "nui_i_main" + +void main() +{ + int nEvent = GetCurrentlyRunningEvent(); + + if (nEvent == EVENT_SCRIPT_MODULE_ON_MODULE_LOAD) + { + SetEventScript(GetModule(), EVENT_SCRIPT_MODULE_ON_NUI_EVENT, "mod_events"); + SetEventScript(GetModule(), EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET, "mod_events"); + + NUI_Initialize(); + NUI_HandleEvents(); + } + else if (nEvent == EVENT_SCRIPT_MODULE_ON_NUI_EVENT) + { + NUI_HandleEvents(); + } + else if (nEvent == EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET) + { + NUI_HandleEvents(GetLastPlayerToSelectTarget()); + } +} diff --git a/_module/nss/mod_nuievent.nss b/_module/nss/mod_nuievent.nss new file mode 100644 index 0000000..f811487 --- /dev/null +++ b/_module/nss/mod_nuievent.nss @@ -0,0 +1,6 @@ +#include "nui_i_main" + +void main() +{ + NUI_HandleEvents(); +} \ No newline at end of file diff --git a/_module/nss/mod_onplaytarget.nss b/_module/nss/mod_onplaytarget.nss new file mode 100644 index 0000000..1536816 --- /dev/null +++ b/_module/nss/mod_onplaytarget.nss @@ -0,0 +1,8 @@ +#include "nui_i_main" + +void main() +{ + ExecuteScript("prc_onplaytarget"); + + NUI_HandleEvents(GetLastPlayerToSelectTarget()); +} \ No newline at end of file diff --git a/_module/nss/nui_c_config.nss b/_module/nss/nui_c_config.nss new file mode 100644 index 0000000..d5707bb --- /dev/null +++ b/_module/nss/nui_c_config.nss @@ -0,0 +1,50 @@ + +/// ---------------------------------------------------------------------------- +/// @file nui_c_config.nss +/// @author Ed Burke (tinygiant98) +/// @brief NUI Form Creation and Management System Configuration +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// FORM INSPECTOR +// ----------------------------------------------------------------------------- + +// The form inspection capability allows you to record data about a form's status, +// including bind values and how various events affect them. To use the inspection +// capability, you must include the `nui_f_inspector` formfile in your module and +// set NUI_FI_RECORD_DATA to the desired value. + +// *** WARNING *** Form inspection is very expensive. Is it recommended that +// you only use the form inspection capability during debug or set this +// value to NUI_FI_WHEN_OPEN to prevent unnecessary traffic. +const int NUI_FI_NEVER = 0; +const int NUI_FI_ALWAYS = 1; +const int NUI_FI_WHEN_OPEN = 2; + +const int NUI_FI_RECORD_DATA = NUI_FI_WHEN_OPEN; + +// ----------------------------------------------------------------------------- +// Custom Functions +// MODIFY THESE TO ENSURE COMPATIBILTY WITH YOUR MODULE'S SYSTEMS +// ----------------------------------------------------------------------------- + +// This system has some basic debugging functionality. In order to make this system work +// under multiple modules without collision, the following function is provided to +// allow you to call whatever debugging system you'd like to call. If you use +// squattingmonk's debug utility, no changes need to be made. The primary system does +// not provide any organic debug calls, but this function is available to all formfiles +// for debugging purposes. +// +const int NUI_DEBUG_SEVERITY_NONE = 0; +const int NUI_DEBUG_SEVERITY_CRITICAL = 1; +const int NUI_DEBUG_SEVERITY_ERROR = 2; +const int NUI_DEBUG_SEVERITY_WARNING = 3; +const int NUI_DEBUG_SEVERITY_NOTICE = 4; +const int NUI_DEBUG_SEVERITY_DEBUG = 5; + +//#include "util_i_debug" + +void NUI_Debug(string sMessage, int nSeverity = 4) +{ + //Debug(sMessage, nSeverity); +} diff --git a/_module/nss/nui_c_storage.nss b/_module/nss/nui_c_storage.nss new file mode 100644 index 0000000..b866661 --- /dev/null +++ b/_module/nss/nui_c_storage.nss @@ -0,0 +1,225 @@ +/// ---------------------------------------------------------------------------- +/// @file nui_c_storage.nss +/// @author Ed Burke (tinygiant98) +/// @brief Persistent Storage formfile configuration settings +/// ---------------------------------------------------------------------------- + +/// @note Most of the following global/default settings can be overriden on a +/// per-container basis by setting a local variable on the container +/// object as noted in the setting descriptions below. Constants, such as +/// PS_TRUE, PS_NONE, etc. should be used in this configuration file, however, +/// actual values must be used when setting local overrides. Those values +/// are provided below. + +/// @warning This system uses player character UUIDs. This can cause issues in +/// single-player modules if the player-character is not exported and the +/// UUID is not saved to the .bic file. + +/// @brief By default, containers will be saved to a database table referenced +/// by the container object's tag. The saved item data includes the UUID +/// and CD Key of the PC that deposited the item so the same container can +/// be used for multiple PCs. To set a specific (unique) name to use as the +/// table name instead, set a local string called `PS_UNIQUE_ID` on the +/// container object and set it to any unique string. + +/// @brief Determines usage of the `Search` button. +/// Configuration File: +/// PS_TRUE to force players to click the `Search` button +/// before an inventory search will commence. This is a good idea +/// for containers that you expect will have large inventories. +/// PS_FALSE to allow real-time searching as the player types +/// characters into the `Search` textbox. +/// Local Override (int): PS_FORCE_SEARCH_BUTTON +/// 1 = PS_TRUE +/// -1 = PS_FALSE +const int PS_FORCE_SEARCH_BUTTON_DEFAULT = PS_FALSE; + +/// @brief Determines whether item object state is saved to the database. The +/// object state includes variables and effects. +/// Configuration File: +/// PS_TRUE saves the object state +/// PS_FALSE does not save the object state +/// Local Override (int): PS_FORCE_OBJECT_STATE +/// 1 = PS_TRUE +/// -1 = PS_FALSE +const int PS_FORCE_OBJECT_STATE_DEFAULT = PS_TRUE; + +/// @brief Sets the item storage limit. +/// Configuration File: +/// PS_UNLIMITED to allow unlimited item storage. +/// Set to any positive integer to limit item storage to that amount. +/// Local Override (int): PS_STORAGE_LIMIT +/// -1 = PS_UNLIMITED +/// Set to any positive integer to limit item storage to that amount. +const int PS_STORAGE_LIMIT_DEFAULT = 500; + +/// @brief Set the maximum distance (meters) a PC can travel from the container +/// before the form will auto-close. +/// Configuration File: +/// PS_UNLIMITED_DISTANCE to never auto-close the form +/// Set to any positive float to limit distance to that amount. +/// Local Override (float): PS_DISTANCE +/// -1.0 = PS_UNLIMITED_DISTANCE +/// Set to any positive float to limit distance to that amount. +const float PS_DISTANCE_DEFAULT = 2.0; + +/// @brief Set the container access type. Container inventories can be accessed +/// by two methods: exclusive and contentious. +/// - Exclusive: Multiple players may open the same container, but each +/// player will only see items they've previously deposited. Players +/// may only withdraw items they've previously deposited. +/// - Contentious: Multiple players may open the same container. All +/// players will see all items deposited by any player. Any player +/// can remove any item, regardless of who originally deposited the +/// item. +/// +/// Configuration File: +/// PS_ACCESS_EXCLUSIVE for exclusive access +/// PS_ACCESS_CONTENTIOUS for contentious access +/// Local Override (int): PS_ACCESS_TYPE +/// 1 = PS_ACCESS_EXCLUSIVE +/// 2 = PS_ACCESS_CONTENTIOUS +const int PS_ACCESS_TYPE_DEFAULT = PS_ACCESS_EXCLUSIVE; + +/// @brief Set the container type. Containers can be of multiple types: +/// - Public: Any player can open, deposit and withdraw items from this +/// container. Whether they are limited to specific items is dependant +/// on the container's access setting. +/// - Character: Creates a 'portable' storage container for any player +/// character. Any container of this type will hold the same inventory +/// for any specific player. +/// - CD Key: Creates a 'portable' storage container for any characters +/// owned by cd-key associated with the player character. Any container +/// of this type will hold the inventory desposited by any character +/// sharing the player's cd key. +/// +/// Configuration File: +/// PS_CONTAINER_PUBLIC for public. +/// PS_CONTAINER_CHARACTER for per-character. +/// PS_CONTAINER_CD_KEY for per-cdkey. +/// Local Override (int): PS_CONTAINER_TYPE +/// 1 = PS_CONTAINER_PUBLIC +/// 2 = PS_CONTAINER_CHARACTER +/// 3 = PS_CONTAINER_CDKEY +const int PS_CONTAINER_TYPE_DEFAULT = PS_CONTAINER_CDKEY; + +/// @brief Set the default container type, if the container is an item. Containers +/// can be of multiple types: +/// - Public: Any player can open, deposit and withdraw items from this +/// container. Whether they are limited to specific items is dependant +/// on the container's access setting. +/// - Character: Creates a 'portable' storage container for any player +/// character. Any container of this type will hold the same inventory +/// for any specific player. +/// - CD Key: Creates a 'portable' storage container for any characters +/// owned by cd-key associated with the player character. Any container +/// of this type will hold the inventory desposited by any character +/// sharing the player's cd key. +/// +/// Configuration File: +/// PS_CONTAINER_PUBLIC for public. +/// PS_CONTAINER_CHARACTER for per-character. +/// PS_CONTAINER_CD_KEY for per-cdkey. +/// Local Override (int): PS_CONTAINER_TYPE +/// 1 = PS_CONTAINER_PUBLIC +/// 2 = PS_CONTAINER_CHARACTER +/// 3 = PS_CONTAINER_CDKEY +const int PS_CONTAINER_ITEM_TYPE_DEFAULT = PS_CONTAINER_CDKEY; + +/// @brief Determines whether the player's inventory window will be opened +/// when a container is opened. +/// Configuration File: +/// PS_TRUE to open the inventory window. +/// PS_FALSE to prevent the window from opening. If the inventory +/// window is already open, this will not close it. +/// Local Override (int): PS_OPEN_INVENTORY +/// 1 = PS_TRUE +/// -1 = PS_FALSE +const int PS_OPEN_INVENTORY_DEFAULT = PS_TRUE; + +/// @brief Determines the maximum amount of gold a container can store. +/// If the container is set to store no gold, the form controls that +/// manipulate gold storage will not be visible on the form. +/// Configuration File: +/// PS_UNLIMITED to allow unlimited gold storage. +/// PS_NONE to prevent gold storage. +/// Set to any positive integer to limit gold to that amount. +/// Local Override (int): PS_MAX_GOLD +/// -1 = PS_UNLIMITED +/// -2 = PS_NONE +/// Set to any positive integer to limit gold to that amount. +const int PS_MAX_GOLD_DEFAULT = 2000000000; + +/// @note Reference these terms for the following option: +/// Container: A persistent storage object in the game, such as a chest. +/// Container Item: An item which: +/// Can have its own inventory +/// Can be carried in a player's inventory + +/// @brief Determines handling for container objects. Containers can optionally +/// store container items. +/// Configuration File: +/// PS_UNLIMITED to allow storage of an unlimited number of container items. +/// PS_NONE to prevent storage of any container items. +/// Set to any positive integer to limit storage to that number of container +/// items. +/// Local Override (int): PS_MAX_CONTAINER_TIMES +/// -1 = PS_UNLIMITED +/// -2 = PS_NONE +/// Set to any positive integer to limit storage to that number of container +/// items. +const int PS_MAX_CONTAINER_ITEMS_DEFAULT = 10; + +/// @brief Determines how many items can be stored in stored container items. +/// Configuration File: +/// PS_UNLIMITED to allow storage of any number of items within a container +/// item's inventory. This will be naturally limited by the size of +/// the container item's inventory. +/// PS_NONE to prevent any items from being stored in a container item. If +/// container item storage is allow and PS_NONE is used, only empty +/// container items can be stored. +/// Set to any positive integer to limit the number of items stored in +/// the inventory of a container item. +/// Local Override (int): PS_MAX_CONTAINER_ITEMS_INVENTORY +/// -1 = PS_UNLIMITED +/// -2 = PS_NONE +/// Set to any positive integer to limit the number of items stored in +/// the inventory of a container item. +/// +/// @note This configuration option has no effect if PS_MAX_CONTAINER_ITEMS_DEFAULT +/// is set to PS_NONE or its local override is set to -1. +/// @warning Items that fail check involved with container item limitations do +/// not have default messages reported to the player. The item will simply fail +/// to be stored. +const int PS_MAX_CONTAINER_ITEMS_INVENTORY_DEFAULT = 100; + +/// @brief Creates the form's title. +/// @param oContainer The container object being used. +/// @param oPC The player using oContainer. +/// @note A local string called `PS_TITLE` may be set on the object for easy +/// reference in this function. The function below is an example and may +/// be modified in any way. The returned value will be displayed as the +/// form's title. +string ps_GetFormTitle(object oContainer, object oPC, int nAccess, int nType) +{ + string sTitle; + if ((sTitle = GetLocalString(oContainer, PS_TITLE)) != "") + return sTitle; + else + { + switch (nType) + { + case PS_CONTAINER_PUBLIC: + return GetTag(oContainer); + case PS_CONTAINER_CDKEY: + return GetName(oPC) + "'s Player-Specific Storage"; + case PS_CONTAINER_CHARACTER: + return GetName(oPC) + "'s Character-Specific Storage"; + } + + if (GetIsPC(OBJECT_SELF)) + return GetName(OBJECT_SELF) + "'s Storage"; + } + + return "Persistent Storage"; +} diff --git a/_module/nss/nui_f_storage.nss b/_module/nss/nui_f_storage.nss new file mode 100644 index 0000000..0dcfc0b --- /dev/null +++ b/_module/nss/nui_f_storage.nss @@ -0,0 +1,1147 @@ +/// ---------------------------------------------------------------------------- +/// @file nui_f_storage.nss +/// @author Ed Burke (tinygiant98) +/// @brief Persistent Storage formfile +/// ---------------------------------------------------------------------------- + +const string FORM_ID = "persistent_storage"; +const string PS_DATABASE = "nui_ps_data"; +const string FORM_VERSION = "0.2.2"; + +const int PS_ACCESS_EXCLUSIVE = 1; +const int PS_ACCESS_CONTENTIOUS = 2; + +const int PS_CONTAINER_PUBLIC = 1; +const int PS_CONTAINER_CHARACTER = 2; +const int PS_CONTAINER_CDKEY = 3; + +const int PS_CONTAINER_ITEMS_NONE = 1; +const int PS_CONTAINER_ITEMS_ANY = 2; + +const int PS_UNLIMITED = -1; +const int PS_NONE = -2; + +const int PS_TRUE = 1; +const int PS_FALSE = -1; + +const float PS_UNLIMITED_DISTANCE = -1.0; + +const string PS_TITLE = "PS_TITLE"; +const string PS_FORCE_SEARCH_BUTTON = "PS_FORCE_SEARCH_BUTTON"; +const string PS_FORCE_OBJECT_STATE = "PS_FORCE_OBJECT_STATE"; +const string PS_STORAGE_LIMIT = "PS_STORAGE_LIMIT"; +const string PS_DISTANCE = "PS_DISTANCE"; +const string PS_UNIQUE_ID = "PS_UNIQUE_ID"; +const string PS_ACCESS_TYPE = "PS_ACCESS_TYPE"; +const string PS_CONTAINER_TYPE = "PS_CONTAINER_TYPE"; +const string PS_OPEN_INVENTORY = "PS_OPEN_INVENTORY"; +const string PS_MAX_GOLD = "PS_MAX_GOLD"; +const string PS_MAX_CONTAINER_ITEMS = "PS_MAX_CONTAINER_ITEMS"; +const string PS_MAX_CONTAINER_ITEMS_INVENTORY = "PS_MAX_CONTAINER_ITEMS_INVENTORY"; +const string PS_ORIGINAL_NAME = "PS_ORIGINAL_NAME"; + +const string PS_DESTROYED = "PS_DESTROYED"; + +const string PS_TARGETING_MODE = "PS_TARGETING_MODE"; +const string PS_SEARCH_STRING = "PS_SEARCH_STRING"; + +const string PS_CONTAINER = "PS_CONTAINER"; + +const string PS_GEOMETRY = "PS_GEOMETRY"; +const string PS_UUID_ARRAY = "PS_UUID_ARRAY"; +const string PS_USERS = "PS_USERS"; + +#include "nui_i_library" +#include "util_i_strings" +#include "nui_c_storage" +#include "util_i_varlists" + +object ps_GetContainer(object oPC) +{ + return GetLocalObject(oPC, PS_CONTAINER); +} + +int ps_GetLocalIntOrDefault(object oPC, string sVarName, int nDefault) +{ + object oContainer = GetIsPC(oPC) ? ps_GetContainer(oPC) : oPC; + int n = GetLocalInt(oContainer, sVarName); + return n ? n : nDefault; +} + +string ps_GetLocalStringOrDefault(object oPC, string sVarName, string sDefault) +{ + object oContainer = GetIsPC(oPC) ? ps_GetContainer(oPC) : oPC; + string s = GetLocalString(oContainer, sVarName); + return s != "" ? s : sDefault; +} + +float ps_GetLocalFloatOrDefault(object oPC, string sVarName, float fDefault) +{ + object oContainer = GetIsPC(oPC) ? ps_GetContainer(oPC) : oPC; + float f = GetLocalFloat(oContainer, sVarName); + return f != 0.0 ? f : fDefault; +} + +string ps_GetContainerID(object oPC) +{ + return ps_GetLocalStringOrDefault(oPC, PS_UNIQUE_ID, GetTag(ps_GetContainer(oPC))); +} + +int ps_GetUseSearchButton(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_FORCE_SEARCH_BUTTON, PS_FORCE_SEARCH_BUTTON_DEFAULT) == PS_TRUE; +} + +int ps_GetSaveObjectState(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_FORCE_OBJECT_STATE, PS_FORCE_OBJECT_STATE_DEFAULT) == PS_TRUE; +} + +int ps_GetContainerType(object oPC) +{ + int nType = GetLocalInt(oPC, PS_CONTAINER_TYPE); + object oContainer = ps_GetContainer(oPC); + + if (!nType && GetObjectType(oContainer) == OBJECT_TYPE_ITEM) + return PS_CONTAINER_ITEM_TYPE_DEFAULT; + else + return PS_CONTAINER_TYPE_DEFAULT; +} + +int ps_GetAccessType(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_ACCESS_TYPE, PS_ACCESS_TYPE_DEFAULT); +} + +int ps_GetMaxItems(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_STORAGE_LIMIT, PS_STORAGE_LIMIT_DEFAULT); +} + +int ps_GetMaxContainerItems(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_MAX_CONTAINER_ITEMS, PS_MAX_CONTAINER_ITEMS_DEFAULT); +} + +int ps_GetMaxContainterItemInventory(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_MAX_CONTAINER_ITEMS_INVENTORY, PS_MAX_CONTAINER_ITEMS_INVENTORY_DEFAULT); +} + +float ps_GetMaxDistance(object oPC) +{ + return ps_GetLocalFloatOrDefault(oPC, PS_DISTANCE, PS_DISTANCE_DEFAULT); +} + +int ps_GetOpenInventory(object oPC) +{ + return ps_GetLocalIntOrDefault(oPC, PS_OPEN_INVENTORY, PS_OPEN_INVENTORY_DEFAULT) == PS_TRUE; +} + +int ps_GetMaxGold(object oPC) +{ + object oContainer = ps_GetContainer(oPC); + if (GetObjectType(oContainer) == OBJECT_TYPE_ITEM) + return PS_NONE; + else + return ps_GetLocalIntOrDefault(oPC, PS_MAX_GOLD, PS_MAX_GOLD_DEFAULT); +} + +object ps_GetFirstUser(object oPC) +{ + return GetListObject(ps_GetContainer(oPC), 0, PS_USERS); +} + +int ps_RemoveUser(object oPC) +{ + return RemoveListObject(ps_GetContainer(oPC), oPC, PS_USERS, TRUE); +} + +string ps_GetOwner(object oPC, string sType = "") +{ + if (ps_GetAccessType(oPC) == PS_ACCESS_CONTENTIOUS && ps_GetContainerType(oPC) != PS_CONTAINER_PUBLIC) + oPC = ps_GetFirstUser(oPC); + + if (sType == "") return GetObjectUUID(oPC) + ":" + GetPCPublicCDKey(oPC, TRUE); + else if (sType == "uuid") return GetObjectUUID(oPC); + else if (sType == "cdkey") return GetPCPublicCDKey(oPC, TRUE); + else return ""; +} + +int ps_GetIsColored(string s) +{ + string sPattern = "***"; + sqlquery sql = SqlPrepareQueryObject(GetModule(), "SELECT @string GLOB @pattern;"); + SqlBindString(sql, "@string", s); + SqlBindString(sql, "@pattern", sPattern); + return SqlStep(sql) ? SqlGetInt(sql, 0) : FALSE; +} + +void ps_BeginTransaction() +{ + SqlStep(SqlPrepareQueryCampaign(PS_DATABASE, "BEGIN TRANSACTION;")); +} + +void ps_CommitTransaction() +{ + SqlStep(SqlPrepareQueryCampaign(PS_DATABASE, "COMMIT TRANSACTION;")); +} + +sqlquery ps_PrepareQuery(string sQuery) +{ + return SqlPrepareQueryCampaign(PS_DATABASE, sQuery); +} + +string ps_GetTableName(object oPC) +{ + return ps_GetContainerID(oPC); +} + +void ps_InitializeDatabase(object oPC) +{ + string sTable = ps_GetTableName(oPC); + string sQuery = "SELECT name FROM sqlite_master WHERE type='table' AND name = @table;"; + sqlquery sql = ps_PrepareQuery(sQuery); + SqlBindString(sql, "@table", sTable); + + if (!SqlStep(sql)) + { + sQuery = + "CREATE TABLE IF NOT EXISTS " + sTable + " (" + + "owner TEXT NOT NULL DEFAULT '', " + + "item_uuid TEXT NOT NULL DEFAULT '', " + + "item_name TEXT NOT NULL DEFAULT '', " + + "item_baseitem INTEGER NOT NULL DEFAULT '', " + + "item_stacksize INTEGER NOT NULL DEFAULT '', " + + "item_iconresref TEXT NOT NULL DEFAULT '', " + + "item_data TEXT_NOT NULL DEFAULT '', " + + "PRIMARY KEY(owner, item_uuid));"; + SqlStep(ps_PrepareQuery(sQuery)); + } +} + +void ps_EnterDepositMode(object oPC) +{ + SetLocalInt(oPC, PS_TARGETING_MODE, TRUE); + EnterTargetingMode(oPC, OBJECT_TYPE_ITEM); +} + +int ps_CountStoredItems(object oPC) +{ + string sAnd, sOwner; + + int nType = ps_GetContainerType(oPC); + if (nType == PS_CONTAINER_CHARACTER || nType == PS_CONTAINER_CDKEY) + { + sAnd = " AND owner GLOB @owner"; + if (nType == PS_CONTAINER_CHARACTER) sOwner = ps_GetOwner(oPC, "uuid") + ":*"; + else sOwner = "*:" + ps_GetOwner(oPC, "cdkey"); + } + + string sQuery = + "SELECT COUNT(*) FROM " + ps_GetTableName(oPC) + " " + + "WHERE item_uuid != 'gold'" + sAnd + ";"; + sqlquery sql = ps_PrepareQuery(sQuery); + + if (sAnd != "") + SqlBindString(sql, "@owner", sOwner); + + return SqlStep(sql) ? SqlGetInt(sql, 0) : 0; +} + +int ps_CountStoredGold(object oPC) +{ + int nType = ps_GetContainerType(oPC); + string sWhere = (nType == PS_CONTAINER_PUBLIC ? "" : " AND owner GLOB @owner"); + + string sQuery = + "SELECT SUM(item_stacksize) FROM " + ps_GetTableName(oPC) + " " + + "WHERE item_uuid == 'gold'" + sWhere + ";"; + sqlquery sql = ps_PrepareQuery(sQuery); + + if (nType == PS_CONTAINER_CHARACTER) SqlBindString(sql, "@owner", ps_GetOwner(oPC, "uuid") + ":*"); + else if (nType == PS_CONTAINER_CDKEY) SqlBindString(sql, "@owner", "*:" + ps_GetOwner(oPC, "cdkey")); + + return SqlStep(sql) ? SqlGetInt(sql, 0) : 0; +} + +string ps_GetIconResref(object oItem, json jItem, int nBaseItem) +{ + if (nBaseItem == BASE_ITEM_CLOAK) + return "iit_cloak"; + else if (nBaseItem == BASE_ITEM_SPELLSCROLL || nBaseItem == BASE_ITEM_ENCHANTED_SCROLL) + { + if (GetItemHasItemProperty(oItem, ITEM_PROPERTY_CAST_SPELL)) + { + itemproperty ip = GetFirstItemProperty(oItem); + while (GetIsItemPropertyValid(ip)) + { + if (GetItemPropertyType(ip) == ITEM_PROPERTY_CAST_SPELL) + return Get2DAString("iprp_spells", "Icon", GetItemPropertySubType(ip)); + + ip = GetNextItemProperty(oItem); + } + } + } + else if (Get2DAString("baseitems", "ModelType", nBaseItem) == "0") + { + json jSimpleModel = JsonPointer(jItem, "/ModelPart1/value"); + if (JsonGetType(jSimpleModel) == JSON_TYPE_INTEGER) + { + string sSimpleModelId = IntToString(JsonGetInt(jSimpleModel)); + while (GetStringLength(sSimpleModelId) < 3) + sSimpleModelId = "0" + sSimpleModelId; + + string sDefaultIcon = Get2DAString("baseitems", "DefaultIcon", nBaseItem); + switch (nBaseItem) + { + case BASE_ITEM_MISCSMALL: + case BASE_ITEM_CRAFTMATERIALSML: + sDefaultIcon = "iit_smlmisc_" + sSimpleModelId; + break; + case BASE_ITEM_MISCMEDIUM: + case BASE_ITEM_CRAFTMATERIALMED: + case 112: + sDefaultIcon = "iit_midmisc_" + sSimpleModelId; + break; + case BASE_ITEM_MISCLARGE: + sDefaultIcon = "iit_talmisc_" + sSimpleModelId; + break; + case BASE_ITEM_MISCTHIN: + sDefaultIcon = "iit_thnmisc_" + sSimpleModelId; + break; + } + + int nLength = GetStringLength(sDefaultIcon); + if (GetSubString(sDefaultIcon, nLength - 4, 1) == "_") + sDefaultIcon = GetStringLeft(sDefaultIcon, nLength - 4); + + string sIcon = sDefaultIcon + "_" + sSimpleModelId; + if (ResManGetAliasFor(sIcon, RESTYPE_TGA) != "") + return sIcon; + } + } + + return Get2DAString("baseitems", "DefaultIcon", nBaseItem); +} + +string ps_FormatGold(int nGold, int nForce = FALSE) +{ + if (nGold < 100000 || nForce) + return FormatInt(nGold, "%,d"); + else if (nGold >= 1000000) + return FormatFloat(nGold / 1000000.0, "%.1fM"); + else + return FormatFloat(nGold / 1000.0, "%.1fk"); +} + +void ps_UpdateGoldBinds(object oPC, int nToken, int nTotal = -1) +{ + if (nTotal == -1) + nTotal = ps_CountStoredGold(oPC); + + NuiSetBind(oPC, nToken, "gold_stored", JsonInt(nTotal)); + NuiSetBind(oPC, nToken, "gold_stored_label", JsonString("Gold: " + ps_FormatGold(nTotal))); + + if (nTotal > 100000) + NuiSetBind(oPC, nToken, "gold_stored_tooltip", JsonString(ps_FormatGold(nTotal, TRUE))); + else + NuiSetBind(oPC, nToken, "gold_stored_tooltip", JsonString("")); + + int nGold = GetGold(oPC); + + NuiSetBind(oPC, nToken, "btn_withdraw_gold", JsonBool(nTotal > 0)); + NuiSetBind(oPC, nToken, "btn_deposit_gold", JsonBool(nGold > 0)); + + int nAmount = StringToInt(JsonGetString(NuiGetBind(oPC, nToken, "gold_amount"))); + int nWithdraw = nAmount <= 0 || nAmount > nTotal ? nTotal : nAmount; + int nDeposit = nAmount <= 0 || nAmount > nGold ? min(ps_GetMaxGold(oPC) - nTotal, nGold) : nAmount; + + NuiSetBind(oPC, nToken, "btn_withdraw_tooltip", JsonString("Withdraw " + ps_FormatGold(nWithdraw, TRUE) + " gold")); + NuiSetBind(oPC, nToken, "btn_deposit_tooltip", JsonString("Deposit " + ps_FormatGold(nDeposit, TRUE) + " gold")); + NuiSetBind(oPC, nToken, "txt_gold_amount", JsonBool(nGold > 0 || nTotal > 0)); +} + +/// @private Updates the amount of gold stored in a container. If the container +/// doesn't allow gold, no action is taken. Stored gold will be updated +/// *by* nGold, not *to* nGold. +/// @param nGold can be positive, negative. `0` is a special case to zero-out +/// the gold in the container (withdrawing all gold) for the owning character +int ps_UpdateGold(object oPC, int nToken, int nGold) +{ + if (ps_GetMaxGold(oPC) <= PS_NONE) return FALSE; + + string sGold = (nGold == 0 ? "@nGold" : "item_stacksize + @nGold"); + string sQuery = + "INSERT INTO " + ps_GetTableName(oPC) + + "(owner, item_uuid, item_baseitem, item_stacksize) " + + "VALUES (@owner, @item_uuid, @item_baseitem, @item_stacksize) " + + "ON CONFLICT (owner, item_uuid) DO UPDATE " + + "SET item_stacksize = " + sGold + ";"; + + sqlquery sql = ps_PrepareQuery(sQuery); + SqlBindString(sql, "@owner", ps_GetOwner(oPC)); + SqlBindString(sql, "@item_uuid", "gold"); + SqlBindInt (sql, "@item_baseitem", BASE_ITEM_GOLD); + SqlBindInt (sql, "@item_stacksize", nGold); + SqlBindInt (sql, "@nGold", nGold); + + SqlStep(sql); + + ps_UpdateGoldBinds(oPC, nToken); + return nGold; +} + +int ps_WithdrawGold(object oPC, int nToken, int nGold) +{ + if (ps_GetMaxGold(oPC) <= PS_NONE) return FALSE; + + if (ps_GetContainerType(oPC) != PS_CONTAINER_PUBLIC) + return ps_UpdateGold(oPC, nToken, -nGold); + + string sTable = ps_GetTableName(oPC); + string sQuery = + "DELETE FROM " + sTable + " WHERE ROWID IN (SELECT ROWID FROM " + sTable + " t1 " + + "WHERE (SELECT SUM(t2.item_stacksize) FROM " + sTable + " t2 WHERE t1.item_stacksize >= " + + "t2.item_stacksize) <= @target ORDER BY t1.item_stacksize ASC) AND item_uuid = 'gold' " + + "RETURNING item_stacksize;"; + sqlquery sql = ps_PrepareQuery(sQuery); + SqlBindInt(sql, "@target", nGold); + + int nRemoved, nRecords; + while (SqlStep(sql)) + { + nRemoved += SqlGetInt(sql, 0); + nRecords++; + } + + if (!nRecords || nGold - nRemoved > 0) + { + sQuery = + "UPDATE " + sTable + " SET item_stacksize = item_stacksize - @gold " + + "WHERE ROWID IN (SELECT ROWID FROM " + sTable + " WHERE item_stacksize >= " + + "@gold AND item_uuid = 'gold' ORDER BY RANDOM() LIMIT 1);"; + sql = ps_PrepareQuery(sQuery); + SqlBindInt(sql, "@gold", nGold - nRemoved); + SqlStep(sql); + } + + return TRUE; +} + +void ps_UpdateItemList(object oPC, int nFlag = FALSE) +{ + /// @note This NUI_DisplaySubform only exists because of an nui issue where shortening a array bound to a listbox + /// a sufficient amount while scrolled near the bottom results in an nui error. To fix this issue, the listbox + /// is implemented as a subform and reloaded here each time this function is called. It effectively fixes the + /// problem, but should be removed when .35 is stable. See additional sections affected by this in DefineForm(). + NUI_DisplaySubform(oPC, FORM_ID, "grpItems", "lstItems"); + + string sAnd, sSearch = GetLocalString(oPC, PS_SEARCH_STRING); + int n, nItems, nGold, nType = ps_GetContainerType(oPC); + int nToken = NUI_GetFormToken(oPC, FORM_ID); + + string sWhere = (nType == PS_CONTAINER_PUBLIC ? "" : " AND owner GLOB @owner"); + json jWhere = JsonArrayInsert(JsonArray(), JsonString(sWhere)); + sWhere += (sSearch == "" ? "" : " AND lower(item_name) GLOB @item"); + jWhere = JsonArrayInsert(jWhere, JsonString(sWhere)); + + string sTable = ps_GetTableName(oPC); + + string sQuery = + "WITH gold AS (SELECT SUM(item_stacksize) pieces FROM " + sTable + " WHERE item_uuid == 'gold'$1 ), " + + "items AS (SELECT item_uuid, IIF(item_stacksize > 1, item_name || ' (x' || item_stacksize || ')', item_name) name, " + + "item_iconresref, json('false') selected FROM " + sTable + " WHERE item_uuid != 'gold'$2 " + + "ORDER BY item_name ASC, item_baseitem ASC) SELECT COUNT(items.item_uuid) items, gold.pieces, " + + "IIF(json_group_array(item_uuid) == json_array(null), json_array(), json_group_array(item_uuid)) uuid, " + + "IIF(json_group_array(name) == json_array(null), json_array(), json_group_array(name)) name, " + + "IIF(json_group_array(item_iconresref) == json_array(null), json_array(), json_group_array(item_iconresref)) resref, " + + "IIF(json_group_array(json(selected)) == json_array(null), json_array(), json_group_array(json(selected))) selected " + + "FROM gold LEFT JOIN items;"; + + sqlquery sql = ps_PrepareQuery(SubstituteString(sQuery, jWhere)); + + if (sSearch != "") SqlBindString(sql, "@item", "*" + GetStringLowerCase(sSearch) + "*"); + + if (nType == PS_CONTAINER_CHARACTER) SqlBindString(sql, "@owner", ps_GetOwner(oPC, "uuid") + ":*"); + else if (nType == PS_CONTAINER_CDKEY) SqlBindString(sql, "@owner", "*:" + ps_GetOwner(oPC, "cdkey")); + + if (SqlStep(sql)) + { + nItems = SqlGetInt(sql, 0); + nGold = SqlGetInt(sql, 1); + SetLocalJson(oPC, PS_UUID_ARRAY, SqlGetJson(sql, 2)); + NuiSetBind(oPC, nToken, "names", SqlGetJson(sql, 3)); + NuiSetBind(oPC, nToken, "icons", SqlGetJson(sql, 4)); + NuiSetBind(oPC, nToken, "selected", SqlGetJson(sql, 5)); + } + + int nMax = ps_GetMaxItems(oPC); + if (nMax >= 0) + { + string sColor; + float f = nItems * 1.0 / nMax; + + if (f > 0.9) sColor = NUI_DefineHexColor(COLOR_RED); + else if (f > 0.75) sColor = NUI_DefineHexColor(COLOR_YELLOW); + else sColor = NUI_DefineHexColor(COLOR_GREEN); + + NuiSetBind(oPC, nToken, "progress", JsonFloat(f)); + NuiSetBind(oPC, nToken, "progress_color", JsonParse(sColor)); + NuiSetBind(oPC, nToken, "progress_tooltip", JsonString(IntToString(nItems) + " of " + IntToString(nMax) + " items stored")); + } + else + { + NuiSetBind(oPC, nToken, "progress", JsonFloat(0.0)); + NuiSetBind(oPC, nToken, "progress_tooltip", JsonString("This container has unlimited item storage")); + } + + NuiSetBind(oPC, nToken, "btn_withdraw", JsonBool(FALSE)); + NuiSetBind(oPC, nToken, "btn_withdraw_all", JsonBool(nItems > 0 && ps_GetAccessType(oPC) != PS_ACCESS_CONTENTIOUS)); + NuiSetBind(oPC, nToken, "btn_deposit", JsonBool(nItems < nMax || nMax <= 0)); + + ps_UpdateGoldBinds(oPC, nToken, nGold); + + if (!nFlag && ps_GetAccessType(oPC) == PS_ACCESS_CONTENTIOUS) + { + object oContainer = ps_GetContainer(oPC); + int n; for (n; n < CountObjectList(oContainer, PS_USERS); n++) + { + object oUser = GetListObject(oContainer, n, PS_USERS); + if (oUser != oPC && NuiFindWindow(oUser, FORM_ID)) + ps_UpdateItemList(oUser, TRUE); + } + } +} + +int ps_CountInventoryItems(object oContainer) +{ + int n; + object oItem = GetFirstItemInInventory(oContainer); + while (GetIsObjectValid(oItem)) + { + n++; + oItem = GetNextItemInInventory(oContainer); + } + + return n; +} + +int ps_CountContainerItems(object oPC) +{ + string sQuery = "SELECT COUNT(*) FROM " + ps_GetTableName(oPC) + " " + + "WHERE item_data != '' AND json_extract(item_data, '$.ItemList') IS NOT NULL;"; + sqlquery sql = ps_PrepareQuery(sQuery); + return SqlStep(sql) ? SqlGetInt(sql, 0) : 0; +} + +int ps_DepositContainerItem(object oPC, object oItem) +{ + if (!GetHasInventory(oItem)) + return TRUE; + + int nMaxContainerItems = ps_GetMaxContainerItems(oPC); + if (nMaxContainerItems <= PS_NONE) + return FALSE; + else + { + if (nMaxContainerItems == PS_UNLIMITED || ps_CountContainerItems(oPC) < nMaxContainerItems) + { + int nMaxItems = ps_GetMaxContainterItemInventory(oPC); + if (nMaxItems == PS_UNLIMITED) + return TRUE; + else + { + if (nMaxItems == PS_NONE) + return !GetIsObjectValid(GetFirstItemInInventory(oItem)); + else + return ps_CountInventoryItems(oItem) <= nMaxItems; + } + } + else + return FALSE; + } +} + +void ps_DepositItem(object oPC, object oItem) +{ + DeleteLocalInt(oPC, PS_TARGETING_MODE); + + if (!GetIsObjectValid(oItem) || GetLocalInt(oItem, PS_DESTROYED) || GetObjectType(oItem) != OBJECT_TYPE_ITEM) + return; + + if (GetItemCursedFlag(oItem) || GetItemPossessor(oItem) != oPC) + return; + + int nStoredItems = ps_CountStoredItems(oPC); + int nMaxItems = ps_GetMaxItems(oPC); + if (nMaxItems > 0 && nStoredItems >= nMaxItems) + { + SendMessageToPC(oPC, "Your storage is full, withdraw an item first."); + return; + } + + if (!ps_DepositContainerItem(oPC, oItem)) + return; + + int nItemBaseItem = GetBaseItemType(oItem); + string sItemName = GetIdentified(oItem) ? GetName(oItem) : GetStringByStrRef(StringToInt(Get2DAString("baseitems", "Name", nItemBaseItem))) + " (Unidentified)"; + json jItemData = ObjectToJson(oItem, ps_GetSaveObjectState(oPC)); + + if (ps_GetIsColored(sItemName)) + { + JsonObjectSet(jItemData, PS_ORIGINAL_NAME, JsonString(sItemName)); + sItemName = UnColorString(sItemName); + } + + string sQuery = + "INSERT INTO " + ps_GetTableName(oPC) + + "(owner, item_uuid, item_name, item_baseitem, item_stacksize, item_iconresref, item_data) " + + "VALUES(@owner, @item_uuid, @item_name ->> '$', @item_baseitem, @item_stacksize, @item_iconresref, @item_data) " + + "RETURNING item_uuid;"; + sqlquery sql = ps_PrepareQuery(sQuery); + + SqlBindString(sql, "@owner", ps_GetOwner(oPC)); + SqlBindString(sql, "@item_uuid", GetObjectUUID(oItem)); + SqlBindJson (sql, "@item_name", JsonString(sItemName)); + SqlBindInt (sql, "@item_baseitem", nItemBaseItem); + SqlBindInt (sql, "@item_stacksize", GetItemStackSize(oItem)); + SqlBindString(sql, "@item_iconresref", ps_GetIconResref(oItem, jItemData, nItemBaseItem)); + SqlBindJson (sql, "@item_data", jItemData); + + if (SqlStep(sql)) + { + if (SqlGetString(sql, 0) == "") + { + NUI_Debug("Error depositing item!", NUI_DEBUG_SEVERITY_ERROR); + return; + } + } + + SetLocalInt(oItem, PS_DESTROYED, TRUE); + DestroyObject(oItem); + + ps_UpdateItemList(oPC); + if (++nStoredItems <= nMaxItems || nMaxItems <= 0) + ps_EnterDepositMode(oPC); +} + +/// @private void version of JsonToObject to prevent mass-withdraw overflow errors. +void ps_JsonToObject(json jObject, location l, object oOwner, int nObjectState) +{ + object oItem = JsonToObject(jObject, l, oOwner, nObjectState); + json jName = JsonObjectGet(jObject, PS_ORIGINAL_NAME); + + if (jName != JsonNull()) + SetName(oItem, JsonGetString(jName)); +} + +void ps_WithdrawItems(object oPC, int nToken, int bForceAll = FALSE) +{ + json jUUIDs = GetLocalJson(oPC, PS_UUID_ARRAY); + if (!JsonGetLength(jUUIDs)) + return; + + string sWhere = (bForceAll ? "" : " WHERE truths.value = true"); + + string sQuery = + "WITH truths AS (SELECT ROWID, value FROM json_each(@truths)), " + + " uuids AS (SELECT rowid, value FROM json_each(@uuids)) " + + "DELETE FROM " + ps_GetTableName(oPC) + " WHERE item_uuid IN " + + "(SELECT uuids.value FROM uuids INNER JOIN truths ON uuids.ROWID " + + "= truths.ROWID" + sWhere + ") RETURNING item_data;"; + + sqlquery sql = ps_PrepareQuery(sQuery); + SqlBindJson(sql, "@uuids", jUUIDs); + SqlBindJson(sql, "@truths", NuiGetBind(oPC, nToken, "selected")); + + ps_BeginTransaction(); + + location l = GetLocation(oPC); + int nState = ps_GetSaveObjectState(oPC); + + while (SqlStep(sql)) + { + json j = SqlGetJson(sql, 0); + DelayCommand(0.0, ps_JsonToObject(j, l, oPC, nState)); + } + + ps_UpdateItemList(oPC); + ps_CommitTransaction(); +} + +void ps_OnFormOpen() +{ + ps_InitializeDatabase(OBJECT_SELF); + + object oContainer = ps_GetContainer(OBJECT_SELF); + int nAccess = ps_GetAccessType(OBJECT_SELF); + int nType = ps_GetContainerType(OBJECT_SELF); + + NUI_SetBind(OBJECT_SELF, FORM_ID, "title", nuiString(ps_GetFormTitle(oContainer, OBJECT_SELF, nAccess, nType))); + + if (ps_GetOpenInventory(OBJECT_SELF)) + PopUpGUIPanel(OBJECT_SELF, GUI_PANEL_INVENTORY); + + ps_UpdateItemList(OBJECT_SELF); +} + +void ps_OnFormClose() +{ + if (ps_GetAccessType(OBJECT_SELF) == PS_ACCESS_CONTENTIOUS) + ps_RemoveUser(OBJECT_SELF); + + DeleteLocalInt(OBJECT_SELF, PS_TARGETING_MODE); + DeleteLocalString(OBJECT_SELF, PS_SEARCH_STRING); + DeleteLocalJson(OBJECT_SELF, PS_UUID_ARRAY); + DeleteLocalObject(OBJECT_SELF, PS_CONTAINER); +} + +void BindForm() +{ + json jBinds = NUI_GetOrphanBinds(FORM_ID); + int n; for (n; n < JsonGetLength(jBinds); n++) + { + string sValue, sBind = JsonGetString(JsonArrayGet(jBinds, n)); + json jValue = JsonNull(); + + if (sBind == "search") sValue = nuiString(""); + + if (sValue != "") + NUI_SetBind(OBJECT_SELF, FORM_ID, sBind, sValue); + else if (jValue != JsonNull()) + NUI_SetBindJ(OBJECT_SELF, FORM_ID, sBind, jValue); + } +} + +void DefineForm() +{ + float fRowHeight = 30.0; + + NUI_CreateForm(FORM_ID, FORM_VERSION); + NUI_SetTOCTitle("Peristent Storage"); + NUI_SetResizable(FALSE); + NUI_SubscribeEvent(EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET); + NUI_SubscribeEvent(EVENT_SCRIPT_PLACEABLE_ON_USED); + NUI_SubscribeEvent(EVENT_SCRIPT_PLACEABLE_ON_OPEN); + NUI_SubscribeEvent(EVENT_SCRIPT_PLACEABLE_ON_CLOSED); + NUI_SubscribeEvent(EVENT_SCRIPT_MODULE_ON_ACTIVATE_ITEM); + { + NUI_AddRow(); + NUI_AddColumn(); + NUI_SetWidth(350.0); + + NUI_AddRow(); + NUI_SetHeight(20.0); + NUI_SetMargin(0.0); + + NUI_AddProgressBar(); + NUI_BindTooltip("progress_tooltip"); + NUI_BindValue("progress"); + NUI_BindForegroundColor("progress_color"); + NUI_CloseRow(); + + NUI_AddRow(); + NUI_SetHeight(fRowHeight); + NUI_SetMargin(0.0); + + NUI_AddTextbox(); + NUI_SetPlaceholder("Search..."); + NUI_BindValue("search", TRUE); + NUI_SetLength(64); + NUI_SetMultiline(FALSE); + NUI_AddCommandButton("btn_clear"); + NUI_SetLabel("X"); + NUI_BindEnabled("btn_clear"); + NUI_SetTooltip("Clear search criteria"); + NUI_SetWidth(35.0); + NUI_AddCommandButton("btn_search"); + NUI_SetLabel("Search"); + NUI_SetWidth(60.0); + NUI_BindEnabled("btn_search"); + NUI_SetTooltip("Search item list by criteria"); + NUI_SetDisabledTooltip("Live search enabled"); + NUI_CloseRow(); + + /// @note Due to the list size issue with NUI in .34, the item listbox had to be + /// implemented as a subform. The commented out section immediately below + /// this AddRow() block is the original implementation and should be reinstanted + /// once (if?) .35 is stable. Additionally, remove the subform definition + /// below. + NUI_AddRow(); + NUI_SetHeight(288.0); + NUI_SetMargin(0.0); + + NUI_AddGroup("grpItems"); + NUI_SetBorder(TRUE); + NUI_SetScrollbars(NUI_SCROLLBARS_NONE); + { + NUI_AddSpacer(); + } NUI_CloseGroup(); + NUI_CloseRow(); + + /* + NUI_AddRow(); + NUI_AddListbox(); + NUI_BindRowCount("icons"); + NUI_SetRowHeight(32.0); + { + NUI_AddGroup(); + NUI_SetBorder(TRUE); + NUI_SetScrollbars(NUI_SCROLLBARS_NONE); + NUI_SetTemplateWidth(32.0); + NUI_SetTemplateVariable(FALSE); + { + NUI_AddImage(); + NUI_BindResref("icons"); + NUI_SetAspect(NUI_ASPECT_FIT); + NUI_SetHorizontalAlignment(NUI_HALIGN_CENTER); + NUI_SetVerticalAlignment(NUI_VALIGN_MIDDLE); + } NUI_CloseGroup(); + NUI_AddCheckbox(); + NUI_BindLabel("names"); + NUI_BindValue("selected"); + } NUI_CloseListbox(); + NUI_CloseRow(); + */ + + NUI_AddRow(); + NUI_SetHeight(fRowHeight); + NUI_SetMargin(0.0); + + NUI_AddCommandButton("btn_withdraw"); + NUI_SetLabel("Withdraw"); + NUI_BindEnabled("btn_withdraw"); + NUI_SetTooltip("Withdraw selected items"); + NUI_SetDisabledTooltip("No items selected for withdrawal"); + NUI_SetWidth(100.0); + NUI_AddCommandButton("btn_withdraw_all"); + NUI_SetLabel("All"); + NUI_BindEnabled("btn_withdraw_all"); + NUI_SetTooltip("Withdraw all items"); + NUI_SetDisabledTooltip("No items available for withdrawal"); + NUI_SetWidth(45.0); + NUI_AddSpacer(); + NUI_AddCommandButton("btn_deposit"); + NUI_SetLabel("Deposit"); + NUI_BindEnabled("btn_deposit"); + NUI_SetTooltip("Select an item to deposit"); + NUI_SetWidth(100.0); + NUI_CloseRow(); + + NUI_AddRow(); + NUI_SetHeight(8.0); + NUI_CloseRow(); + + NUI_AddRow(); + NUI_SetHeight(fRowHeight); + NUI_BindVisible("showGold"); + NUI_SetMargin(0.0); + + NUI_AddGroup(); + NUI_SetBorder(TRUE); + NUI_SetScrollbars(NUI_SCROLLBARS_NONE); + NUI_SetWidth(100.0); + NUI_BindTooltip("gold_stored_tooltip"); + { + NUI_AddLabel(); + NUI_BindLabel("gold_stored_label"); + NUI_SetForegroundColor(NUI_DefineHexColor(COLOR_GOLD)); + NUI_SetHorizontalAlignment(NUI_HALIGN_LEFT); + NUI_SetVerticalAlignment(NUI_VALIGN_MIDDLE); + } NUI_CloseGroup(); + + NUI_AddTextbox(); + NUI_SetPlaceholder("Amount..."); + NUI_BindEnabled("txt_gold_amount"); + NUI_BindValue("gold_amount", TRUE); + NUI_BindTooltip("gold_amount_tooltip"); + NUI_SetLength(64); + NUI_SetMultiline(FALSE); + + NUI_AddCommandButton("btn_withdraw_gold"); + NUI_SetLabel("Withdraw"); + NUI_BindEnabled("btn_withdraw_gold"); + NUI_BindTooltip("btn_withdraw_tooltip"); + NUI_SetWidth(75.0); + + NUI_AddCommandButton("btn_deposit_gold"); + NUI_SetLabel("Deposit"); + NUI_BindEnabled("btn_deposit_gold"); + NUI_BindTooltip("btn_deposit_tooltip"); + NUI_SetWidth(75.0); + NUI_CloseRow(); + NUI_CloseColumn(); + NUI_CloseRow(); + } + + /// @note Remove this subform definition when .35 is stable. + NUI_CreateSubform("lstItems"); + { + NUI_AddRow(); + NUI_AddListbox(); + NUI_BindRowCount("icons"); + NUI_SetRowHeight(32.0); + NUI_SetBorder(FALSE); + { + NUI_AddGroup(); + NUI_SetBorder(TRUE); + NUI_SetScrollbars(NUI_SCROLLBARS_NONE); + NUI_SetTemplateWidth(32.0); + NUI_SetTemplateVariable(FALSE); + { + NUI_AddImage(); + NUI_BindResref("icons"); + NUI_SetAspect(NUI_ASPECT_FIT); + NUI_SetHorizontalAlignment(NUI_HALIGN_CENTER); + NUI_SetVerticalAlignment(NUI_VALIGN_MIDDLE); + } NUI_CloseGroup(); + NUI_AddCheckbox(); + NUI_BindLabel("names"); + NUI_BindValue("selected", TRUE); + } NUI_CloseListbox(); + NUI_CloseRow(); + } + + NUI_CreateDefaultProfile(); + { + NUI_SetProfileBind("geometry", NUI_DefineRectangle(360.0, 0.0, 370.0, 470.0)); + NUI_SetProfileBind("search", nuiString("")); + NUI_SetProfileBind("showGold", nuiBool(TRUE)); + NUI_SetProfileBind("showDMPanel", nuiBool(FALSE)); + } + + NUI_CreateProfile("noGold"); + { + NUI_SetProfileBind("showGold", nuiBool(FALSE)); + } +} + +void HandleNUIEvents() +{ + struct NUIEventData ed = NUI_GetEventData(); + + if (ed.sEvent == "click") + { + if (ed.sControlID == "btn_withdraw") + ps_WithdrawItems(ed.oPC, ed.nToken); + else if (ed.sControlID == "btn_withdraw_all") + ps_WithdrawItems(ed.oPC, ed.nToken, TRUE); + else if (ed.sControlID == "btn_deposit") + ps_EnterDepositMode(ed.oPC); + else if (ed.sControlID == "btn_deposit_gold") + { + int nAmount, nGold = GetGold(ed.oPC); + string sAmount = JsonGetString(NuiGetBind(ed.oPC, ed.nToken, "gold_amount")); + + if (sAmount == "") + nAmount = nGold; + else + nAmount = clamp(StringToInt(sAmount), 0, nGold); + + if ((nGold = ps_GetMaxGold(ed.oPC)) > -2) + nAmount = min(nAmount, nGold - JsonGetInt(NuiGetBind(ed.oPC, ed.nToken, "gold_stored"))); + + if (nAmount <= 0) return; + + if (ps_UpdateGold(ed.oPC, ed.nToken, nAmount)) + AssignCommand(ed.oPC, TakeGoldFromCreature(nAmount, ed.oPC, TRUE)); + + DelayCommand(0.1, ps_UpdateGoldBinds(ed.oPC, ed.nToken)); + } + else if (ed.sControlID == "btn_withdraw_gold") + { + int nAmount, nGold = JsonGetInt(NuiGetBind(ed.oPC, ed.nToken, "gold_stored")); + string sAmount = JsonGetString(NuiGetBind(ed.oPC, ed.nToken, "gold_amount")); + + if (sAmount == "") + nAmount = nGold; + else + nAmount = clamp(StringToInt(sAmount), 0, nGold); + + if (nAmount <= 0) return; + + if (ps_WithdrawGold(ed.oPC, ed.nToken, nAmount)) + GiveGoldToCreature(ed.oPC, nAmount); + + ps_UpdateGoldBinds(ed.oPC, ed.nToken); + } + else if (ed.sControlID == "btn_search") + { + SetLocalString(ed.oPC, PS_SEARCH_STRING, JsonGetString(NuiGetBind(ed.oPC, ed.nToken, "search"))); + ps_UpdateItemList(ed.oPC); + } + else if (ed.sControlID == "btn_clear") + { + NuiSetBind(ed.oPC, ed.nToken, "search", JsonString("")); + + if (ps_GetUseSearchButton(ed.oPC)) + { + DeleteLocalString(ed.oPC, PS_SEARCH_STRING); + ps_UpdateItemList(ed.oPC); + } + } + } + else if (ed.sEvent == "watch") + { + if (ed.sControlID == "geometry") + { + SetLocalJson(ed.oPC, PS_GEOMETRY, NuiGetBind(ed.oPC, ed.nToken, "geometry")); + } + else if (ed.sControlID == "selected") + { + json jSelected = JsonFind(NuiGetBind(ed.oPC, ed.nToken, ed.sControlID), JsonBool(TRUE)); + NuiSetBind(ed.oPC, ed.nToken, "btn_withdraw", JsonBool(jSelected != JsonNull())); + } + else if (ed.sControlID == "search") + { + string sSearch = JsonGetString(NuiGetBind(ed.oPC, ed.nToken, "search")); + int nSearch = GetStringLength(sSearch); + + NuiSetBind(ed.oPC, ed.nToken, "btn_clear", JsonBool(nSearch > 0)); + + if (ps_GetUseSearchButton(ed.oPC)) + { + NuiSetBind(ed.oPC, ed.nToken, "btn_search", JsonBool(nSearch > 0)); + if (!nSearch) + { + DeleteLocalString(ed.oPC, PS_SEARCH_STRING); + ps_UpdateItemList(ed.oPC); + } + } + else + { + NuiSetBind(ed.oPC, ed.nToken, "btn_search", JsonBool(FALSE)); + SetLocalString(ed.oPC, PS_SEARCH_STRING, sSearch); + ps_UpdateItemList(ed.oPC); + } + } + else if (ed.sControlID == "gold_amount") + { + string sAmount = JsonGetString(NuiGetBind(ed.oPC, ed.nToken, ed.sControlID)); + if (!GetIsNumeric(sAmount)) + { + NuiSetBind(ed.oPC, ed.nToken, "btn_withdraw_gold", jFalse); + NuiSetBind(ed.oPC, ed.nToken, "btn_deposit_gold", jFalse); + NuiSetBind(ed.oPC, ed.nToken, "gold_amount_tooltip", JsonString("Error: Only Digits Allowed")); + return; + } + + NuiSetBind(ed.oPC, ed.nToken, "gold_amount_tooltip", JsonString("")); + + int nStored = JsonGetInt(NuiGetBind(ed.oPC, ed.nToken, "gold_stored")); + int nAmount = clamp(StringToInt(sAmount), 0, max(nStored, GetGold(ed.oPC))); + + if (StringToInt(sAmount) > nAmount) + NuiSetBind(ed.oPC, ed.nToken, ed.sControlID, JsonString(IntToString(nAmount))); + + ps_UpdateGoldBinds(ed.oPC, ed.nToken); + } + } + else if (ed.sEvent == "open") + ps_OnFormOpen(); + else if (ed.sEvent == "close") + ps_OnFormClose(); +} + +void ps_CloseContainer(object oPC) +{ + if (NUI_GetFormToken(oPC, FORM_ID)) + NUI_CloseForm(oPC, FORM_ID); +} + +/// @private Closes the form if the player moves too far away from the container. +void ps_OnPCHeartbeat(object oPC, object oContainer) +{ + object oLastContainer = GetLocalObject(oPC, PS_CONTAINER); + if (oLastContainer == OBJECT_INVALID || oLastContainer != oContainer) + return; + + if (GetObjectType(oContainer) == OBJECT_TYPE_ITEM) + return; + + float fMax = ps_GetMaxDistance(oPC); + if (fMax < 0.0) return; + + if (GetDistanceBetween(oLastContainer, oPC) > fMax) + ps_CloseContainer(oPC); + else + AssignCommand(oPC, DelayCommand(2.0, ps_OnPCHeartbeat(oPC, oLastContainer))); +} + +/// @private Closes the form for any using player that moves too far away from the +/// container. +void ps_OnContainerHeartbeat(object oContainer) +{ + if (GetObjectType(oContainer) == OBJECT_TYPE_ITEM) + return; + + float fMax = ps_GetLocalFloatOrDefault(oContainer, PS_DISTANCE, PS_DISTANCE_DEFAULT); + if (fMax < 0.0) return; + + int nAccess = ps_GetLocalIntOrDefault(oContainer, PS_ACCESS_TYPE, PS_ACCESS_TYPE_DEFAULT); + if (nAccess == PS_ACCESS_CONTENTIOUS) + { + int n; for (n; n < CountObjectList(oContainer, PS_USERS); n++) + { + object oPC = GetListObject(oContainer, n, PS_USERS); + if (GetDistanceBetween(oContainer, oPC) > fMax) + ps_CloseContainer(oPC); + } + } + + if (CountObjectList(oContainer, PS_USERS)) + AssignCommand(oContainer, DelayCommand(2.0, ps_OnContainerHeartbeat(oContainer))); +} + +void ps_OpenContainer(object oPC, object oContainer = OBJECT_INVALID) +{ + //ps_OnFormClose(); + + if (oContainer == OBJECT_INVALID) + oContainer = oPC == OBJECT_SELF ? GetLocalObject(oPC, NUI_OBJECT) : OBJECT_SELF; + SetLocalObject(oPC, PS_CONTAINER, oContainer); + + if (ps_GetAccessType(oPC) == PS_ACCESS_CONTENTIOUS) + AddListObject(oContainer, oPC, PS_USERS, TRUE); + + if (GetObjectType(oContainer) != OBJECT_TYPE_ITEM) + { + DelayCommand(2.0, ps_OnPCHeartbeat(oPC, oContainer)); + DelayCommand(2.0, ps_OnContainerHeartbeat(oContainer)); + } + + NUI_DisplayForm(oPC, FORM_ID, ps_GetMaxGold(oPC) > -2 ? "default" : "noGold"); +} + +void HandleModuleEvents() +{ + object oPC = OBJECT_SELF; + + switch (GetCurrentlyRunningEvent()) + { + case EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET: + { + object oPC = GetLastPlayerToSelectTarget(); + + if (!GetLocalInt(oPC, PS_TARGETING_MODE) || NuiFindWindow(oPC, FORM_ID) == 0) + return; + + ps_DepositItem(oPC, GetTargetingModeSelectedObject()); + } break; + case EVENT_SCRIPT_PLACEABLE_ON_USED: + if (!GetIsPC(oPC)) oPC = GetLastUsedBy(); + case EVENT_SCRIPT_PLACEABLE_ON_OPEN: + if (!GetIsPC(oPC)) oPC = GetLastOpenedBy(); + ps_OpenContainer(oPC); + break; + case EVENT_SCRIPT_PLACEABLE_ON_CLOSED: + if (!GetIsPC(OBJECT_SELF)) oPC = GetLastClosedBy(); + ps_CloseContainer(oPC); + break; + case EVENT_SCRIPT_MODULE_ON_ACTIVATE_ITEM: + ps_OpenContainer(GetItemActivator(), GetItemActivated()); + break; + default: + } +} diff --git a/_module/nss/nui_i_library.nss b/_module/nss/nui_i_library.nss new file mode 100644 index 0000000..559da58 --- /dev/null +++ b/_module/nss/nui_i_library.nss @@ -0,0 +1,43 @@ +/// ---------------------------------------------------------------------------- +/// @file nui_i_library.nss +/// @author Ed Burke (tinygiant98) +/// @brief Boilerplate code for creating a library dispatcher. Should only be +/// included in library scripts as it implements main(). +/// ---------------------------------------------------------------------------- + +#include "nui_i_main" + +// ----------------------------------------------------------------------------- +// Function Protoypes +// ----------------------------------------------------------------------------- + +void DefineForm(); +void BindForm(); +void HandleNUIEvents(); +void HandleModuleEvents(); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +// These are dummy implementations to prevent nwnsc from complaining that they +// do not exist. If you want to compile in the toolset rather than using nwnsc, +// comment these lines out. +//#pragma default_function(DefineForm) +//#pragma default_function(BindForm) +//#pragma default_function(HandleNUIEvents) +//#pragma default_function(HandleModuleEvents) + +// ----------------------------------------------------------------------------- +// Library Dispatch +// ----------------------------------------------------------------------------- + +void main() +{ + string sOperation = GetScriptParam("NUI_FUNCTION"); + + if (sOperation == NUI_DEFINE) DefineForm(); + else if (sOperation == NUI_BIND) BindForm(); + else if (sOperation == NUI_EVENT_NUI) HandleNUIEvents(); + else HandleModuleEvents(); +} diff --git a/_module/nss/nui_i_main.nss b/_module/nss/nui_i_main.nss new file mode 100644 index 0000000..345014a --- /dev/null +++ b/_module/nss/nui_i_main.nss @@ -0,0 +1,2697 @@ +/// ---------------------------------------------------------------------------- +/// @file nui_i_main.nss +/// @author Ed Burke (tinygiant98) +/// @brief NUI Form Creation and Management System +/// ---------------------------------------------------------------------------- + +#include "util_i_csvlists" +#include "util_i_color" +#include "nui_c_config" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string NUI_VERSION = "0.3.0"; +const string NUI_DATABASE = "nui_form_data"; + +const int NUI_ORIENTATION_ROW = 0; +const int NUI_ORIENTATION_COLUMN = 1; + +const int NUI_DRAW_ABOVE = 1; +const int NUI_DRAW_BELOW = -1; + +const int NUI_DRAW_ALWAYS = 0; +const int NUI_DRAW_MOUSEOFF = 1; +const int NUI_DRAW_MOUSEHOVER = 2; +const int NUI_DRAW_MOUSELEFT = 3; +const int NUI_DRAW_MOUSERIGHT = 4; +const int NUI_DRAW_MOUSEMIDDLE = 5; + +const int NUI_SCROLLBARS_NONE = 0; +const int NUI_SCROLLBARS_X = 1; +const int NUI_SCROLLBARS_Y = 2; +const int NUI_SCROLLBARS_BOTH = 3; +const int NUI_SCROLLBARS_AUTO = 4; + +const int NUI_CHART_LINE = 0; +const int NUI_CHART_BAR = 1; + +const int NUI_ASPECT_FIT = 0; +const int NUI_ASPECT_FILL = 1; +const int NUI_ASPECT_FIT100 = 2; +const int NUI_ASPECT_EXACT = 3; +const int NUI_ASPECT_EXACTSCALED = 4; +const int NUI_ASPECT_STRETCH = 5; + +const int NUI_HALIGN_CENTER = 0; +const int NUI_HALIGN_LEFT = 1; +const int NUI_HALIGN_RIGHT = 2; + +const int NUI_VALIGN_MIDDLE = 0; +const int NUI_VALIGN_TOP = 1; +const int NUI_VALIGN_BOTTOM = 2; + +const string NUI_DEFINE = "DefineForm"; +const string NUI_BIND = "BindForm"; +const string NUI_EVENT_NUI = "HandleNUIEvents"; +const string NUI_EVENT_MOD = "HandleModuleEvents"; + +const string NUI_OBJECT = "NUI_OBJECT"; + +const int NUI_FI_EVENT_UPDATE_FORMS = 100001; +const int NUI_FI_EVENT_UPDATE_EVENTS = 100002; + +json jTrue = JsonBool(TRUE); +json jFalse = JsonBool(FALSE); + +// TODO remove upon debug completion +const int NUI_USE_CAMPAIGN_DATABASE = TRUE; +const string NUI_FORMFILE_PREFIX = "nui_f_"; + +struct NUIEventData { + object oPC; // PC object interacting with the form + int nToken; // Subject form token + string sFormID; // Form ID as assigned during the form definition process + string sEvent; // Event - mouseup, click, etc. + string sControlID; // Control ID as assigned during the form definition process + int nIndex; // Index of control in array, if the control is in an array (listbox) + json jPayload; // Event payload, if it exists +}; + +// ----------------------------------------------------------------------------- +// NUI/JSON Helpers +// ----------------------------------------------------------------------------- + +/// @brief Formats a json-parseable string. +/// @param s String value. +string nuiString(string s); + +/// @brief Formats a json-parseable integer. +/// @param n Integer value. +string nuiInt(int n); + +/// @brief Formats a json-parseable float. +/// @param f Float value. +string nuiFloat(float f); + +/// @brief Formats a json-parseable boolean. +/// @param b Boolean value. +string nuiBool(int b); + +/// @brief Creates a json-parseable object for a data bind. +/// @param sBind Bind variable. +/// @param bWatch TRUE to set a bind watch. +string nuiBind(string sBind, int bWatch = FALSE); + +/// @brief Creates a json-parseable object for referencing strings by StringRef. +/// @param nStrRef StringRef value. +string nuiStrRef(int nStrRef); + +/// @brief Creates a json-parseable object for a null value. +string nuiNull(); + +// ----------------------------------------------------------------------------- +// Form Definition, Controls, Custom JSON Structures, Drawing Elements +// ----------------------------------------------------------------------------- + +/// @brief Must be called during the module load process. Initializes the +/// required nui database tables and loads all available formfiles. +void NUI_Initialize(); + +/// @brief Creates a form template with all required form properties set to +/// default values: +/// accepts_input: true +/// border: true +/// closable: true +/// collapsible: true +/// geometry: bind:"geometry" +/// resizable: true +/// title: bind:"title" +/// transparent: false +/// @param sID FormID. +/// @param sVersion A local version set into the form's json structure as +/// "local_version". This is different than the nui system's required +/// version, which should never be changed. +void NUI_CreateForm(string sID, string sVersion = ""); + +/// @brief Define a subform. +/// @param sID Subform ID. +/// @warning Must be used only during the form definition process and after +/// definition of the main form. +void NUI_CreateSubform(string sID); + +/// @brief Define a point based on a single coordinate set. +/// @param x X-coordinate. +/// @param y Y-coordinate. +/// @returns A json-parseable string representing a single coordinate set. +/// {"x":x.x, "y":y.y} +string NUI_DefinePoint(float x, float y); + +/// @brief Get an array of line endpoint coordinates. +/// @param x1 Start point x-coordinate. +/// @param y1 Start point y-coordinate. +/// @param x2 End point x-coordinate. +/// @param y2 End point y-coordinate. +/// @returns A json-parseable string representing an array of coordinates +/// that can be used in NUI_DrawLine(). +/// [x1, y1, x2, y2] +string NUI_GetLinePoints(float x1, float y1, float x2, float y2); + +/// @brief Add a single coordinate set to an empty or existing coordinate array. +/// @param sPoints Coordinate array. Can be a pre-existing arry as created by +/// NUI_GetLinePoints, an empty array string ("[]"), or an empty string (""). +/// @param x X-coordinate. +/// @param y Y-coordinate. +/// @returns A json-parseable string representing an array of coordinates +/// that can be used in NUI_DrawLine(). +/// [..., x, y] +string NUI_AddLinePoint(string sPoints, float x, float y); + +/// @brief Define an nui-usable color vector via rgba values. +/// @param r Red value. +/// @param g Green value. +/// @param b Blue value. +/// @param a Transparency value. +/// @returns A json-parseable string representing a color. +/// {"r":r, "g":g, "b":b, "a":a} +string NUI_DefineRGBColor(int r, int g, int b, int a = 255); + +/// @brief Define an nui-usable color vector via hsv values. +/// @param h Hue. +/// @param s Saturation. +/// @param v Value. +/// @returns A json-parseable string representing a color. +/// {"r":r, "g":g, "b":b, "a":a} +string NUI_DefineHSVColor(float h, float s, float v); + +/// @brief Define an nui-usable color vector via hex value. +/// @param nColor Hex color. +/// @returns A json-parseable string representing a color. +/// {"r":r, "g":g, "b":b, "a":a} +string NUI_DefineHexColor(int nColor); + +/// @brief Define a random nui-usable color vector. +/// @returns A json-parseable string representing a color. +/// {"r":r, "g":g, "b":b, "a":a} +string NUI_DefineRandomColor(); + +/// @brief Define a rectangle based on coordinates and dimensions. +/// @param x X-coordinate, top left corner. +/// @param y Y-coordinate, top left corner. +/// @param w Width. +/// @param h Height. +/// @returns A json-parseable string representing a rectangular region. +/// {"x":x, "y":y, "w":w, "h":h} +string NUI_DefineRectangle(float x, float y, float w, float h); + +/// @brief Get an array of rectangle corner coordinates. +/// @param x X-coordinate, top left corner. +/// @param y Y-coordinate, top left corner. +/// @param w Width. +/// @param h Height. +/// @returns A json-parseable string representing an array of coordinates +/// that can be used in NUI_DrawLine. +/// [x, y, x+w, y, x+w, y+h, x, y+h, x, y] +string NUI_GetRectanglePoints(float x, float y, float w, float h); + +/// @brief Get an array of rectangle corner coordinates. +/// @param sRectangle A json-parseable string representing a rectangular +/// region as returned by NUI_DefineRectangle(). +/// @returns A json-parseable string representing an array of coordinates +/// that can be used in NUI_DrawLine. +/// [x, y, x+w, y, x+w, y+h, x, y+h, x, y] +string NUI_GetDefinedRectanglePoints(string sRectangle); + +/// @brief Define a circle based on coordinates and radius. +/// @param x X-coordinate, center point. +/// @param y Y-coordinate, center point. +/// @param r Radius. +/// @returns A json-parseable string representing a rectangular region +/// that can be used in NUI_DrawDefinedCircle(). +string NUI_DefineCircle(float x, float y, float r); + +/// @brief Add a column to the form or control group. +/// @param fWidth Column width. If omitted, width will calculated automatically. +/// @note Definition must be closed with NUI_CloseColumn(). +void NUI_AddColumn(float fWidth = -1.0); + +/// @brief Close a column definition. +void NUI_CloseColumn(); + +/// @brief Add a row to the form or control group. +/// @param fWidth Row height. If omitted, height will calculated automatically. +/// @note Definition must be closed with NUI_CloseRow(). +void NUI_AddRow(float fHeight = -1.0); + +/// @brief Close a row definition. +void NUI_CloseRow(); + +/// @brief Add a control group to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note Control group can contain other controls and can act as a layout element +/// for tab controls or as a form's subregion, containing its own columns and rows +/// of elements. Definition must be closed with NUI_CloseGroup(). +void NUI_AddGroup(string sID = ""); + +/// @brief Close a control group definition. +void NUI_CloseGroup(); + +/// @brief Add a chart to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddChart(string sID = ""); + +/// @brief Add a checkbox to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddCheckbox(string sID = ""); + +/// @brief Add a color picker to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddColorPicker(string sID = ""); + +/// @brief Add a combobox to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddCombobox(string sID = ""); + +/// @brief Add a command button to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddCommandButton(string sID = ""); + +/// @brief Add a float-based slider to the form or control group +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddFloatSlider(string sID = ""); + +/// @brief Add an image to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddImage(string sID = ""); + +/// @brief Add an image button to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddImageButton(string sID = ""); + +/// @brief Add an int-based slider to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddIntSlider(string sID = ""); + +/// @brief Add a label to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddLabel(string sID = ""); + +/// @brief Add a listbox to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note Each control added to the listbox row template can have individual +/// properties set. Additionally, template element properties can be set +/// with NUI_SetTemplateVariable() and NUI_SetTemplateWidth(). Definition +/// must be closed with NUI_CloseListbox(). +void NUI_AddListbox(string sID = ""); + +/// @brief Close a listbox definition. +void NUI_CloseListbox(); + +/// @brief Add an option group to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note Radio buttons may be added with NUI_SetElements(). Option group +/// values are 0-based, starting with the first radio button added. +void NUI_AddOptionGroup(string sID = ""); + +/// @brief Add a progress bar to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddProgressBar(string sID = ""); + +/// @brief Add a spacer to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +void NUI_AddSpacer(string sID = ""); + +/// @brief Add an editable textbox to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note To force the textbox to be non-editable, use NUI_SetStatic(); +void NUI_AddTextbox(string sID = ""); + +/// @brief Add a toggle button to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note Toggle buttons may be added with NUI_AddElement() or +/// NUI_AddElement(). Option group values are 0-based, starting with +/// the first radio button added. +void NUI_AddToggleButton(string sID = ""); + +/// @brief Add an toggle (option) button group to the form or control group. +/// @param sID Control id. Returned by NuiGetEventElement() (nwscript) or as +/// ed.sControlID in event data. +/// @note Toggle buttons may be added with NUI_SetElements(). Option group +/// values are 0-based, starting with the first toggle button added. +void NUI_AddToggleGroup(string sID = ""); + +/// @brief Add a canvas to a control or control group. +/// @note Any number of drawlist elements may be added to a single canvas. +/// Drawlist elements can be added to any control or control group, but +/// cannot be added to the base form. Definitions must be closed with +/// NUI_CloseCanvas(). +void NUI_AddCanvas(); + +/// @brief Close a canvas definition. +void NUI_CloseCanvas(); + +/// @brief Draw a line or polyline on the canvas. +/// @param sPoints Json-parseable points array as defined by NUI_GetLinePoints(), +/// NUI_AddLinePoint(), or NUI_GetRectanglePoints(). +/// @note Line point coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawLine(string sPoints); + +/// @brief Draw a rectangle on the canvas. +/// @param x X-coordinate, top left corner. +/// @param y Y-coordinate, top left corner. +/// @param w Width. +/// @param h Height. +/// @note X and Y coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawRectangle(float x, float y, float w, float h); + +/// @brief Draw a rectangle on the canvas. +/// @param sRect Json-parseable string representing a rectangular region as defined +/// by NUI_DefineRectangle(). +/// @note X and Y coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawDefinedRectangle(string sRect); + +/// @brief Draw a circle on the canvas. +/// @param x X-coordinate, center point. +/// @param y Y-coordinate, center point. +/// @param r Radius. +/// @note X and Y coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawCircle(float x, float y, float r); + +/// @brief Draw a circle on the canvas. +/// @param sCircle Json-parseable string representing a rectangular region as defined +/// by NUI_DefineCircle(). +/// @note X and Y coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawDefinedCircle(string sCircle); + +/// @brief Draw a textbox on the canvas. +/// @param sRect Json-parseable string representing a rectanglur region as defined +/// by NUI_DefineRectangle(). +/// @param sText Text to be displayed in the drawn textbox. +/// @note X and Y coordinates are relative to the top-left corner of the control +/// the drawlist element is being added to. +void NUI_DrawText(string sRect, string sText); + +/// @brief Draw an image on the canvas. +/// @param sResref Resref of the image to be displayed. +/// @param sRect Json-parseable string representing a rectanglur region as defined +/// by NUI_DefineRectangle(). +/// @param nAspect Aspect ratio. +/// @param nHAlign Horizontal Alignment. +/// @param nValign Vertical Alignment. +void NUI_DrawImage(string sResref, string sRect, int nAspect, int nHAlign, int nVAlign); + +/// @brief Draw an arc on the canvas. +/// @param sCenter Json-parseable string representing the center of the arc as defined by +/// NUI_DefinePoint(). +/// @param r Radius. +/// @param fAMin Start angle. +/// @param fAMax End angle. +/// @note fAMin and fAMax are measured in fractions of PI radians from the start radian of +/// (0 * PI) which is visually represented as 090 degrees. A complete circle is 2 * PI. +/// To draw a 90 degree arc from 180 degrees to 270 degrees, use +/// NUI_DrawArc([sCenter], [r], 0.5 * PI, PI);. The radius arms will also be drawn. +/// @note To achieve any given angle with 0 degrees as straight up, use +/// fAMin = (-0.5 * PI) + ( / 180.0) * PI; +void NUI_DrawArc(string sCenter, float r, float fAMin, float fAMax); + +/// @brief Draw a bezier curve on the canvas. +/// @param sStart Json-parseable string representing the curve's start point. +/// @param sEnd Json-parseable string representing the curve's end point. +/// @param sCtrl0 Json-parseable string representing the curve's first control point. +/// @param sCtrl1 Json-parseable string representing the curve's second control point. +/// @note All arguments for this function can be defined by NUI_DefinePoint(). +/// @note More information about bezier curves: https://en.wikipedia.org/wiki/B%C3%A9zier_curve +void NUI_DrawCurve(string sStart, string sEnd, string sCtrl0, string sCtrl1); + +/// @brief Binds the drawlist element's points property. +/// @param sBind Variable to bind. +void NUI_BindLine(string sBind); + +/// @brief Binds the drawlist element's rectangle property. +/// @param sBind Variable to bind. +void NUI_BindCircle(string sBind); + +/// @brief Binds the drawlist element's text and rectangle properties. +/// @param sRectangle Variable to bind. +/// @param sText Variable to bind. +void NUI_BindTextbox(string sRectangle, string sText); + +/// @brief Binds the drawlist element's rectangle, resref (image), aspect and alignment properties. +/// @param sResref Variable to bind. +/// @param sRectangle Variable to bind. +/// @param sAspect Variable to bind. +/// @param sHAlign Variable to bind. +/// @param sVAlign Variable to bind. +void NUI_BindImage(string sResref, string sRectangle, string sAspect, string sHAlign, string sVAlign); + +/// @brief Binds the drawlist element's center, radius, start angle and end angle properties. +/// @param sCenter Variable to bind. +/// @param sRadius Variable to bind. +/// @param sStartAngle Variable to bind. +/// @param sEndAngle Variable to bind. +void NUI_BindArc(string sCenter, string sRadius, string sStartAngle, string sEndAngle); + +/// @brief Binds the drawlist element's start, end, and control point properties. +/// @param sStart Variable to bind. +/// @param sEnd Variable to bind. +/// @param sCtrl0 Variable to bind. +/// @param sCtrl1 Variable to bind. +void NUI_BindCurve(string sStart, string sEnd, string sCtrl0, string sCtrl1); + +/// @brief Binds the form's accepts input property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindAcceptsInput(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's aspect property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindAspect(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's or control's border property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindBorder(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's closable property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindClosable(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's collapsible property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindCollapsible(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's color property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindColor(string sBind, int bWatch = FALSE); + +/// @brief Binds the disabled control's tooltip property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindDisabledTooltip(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's elements property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindElements(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's enabled property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindEnabled(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's encouraged property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindEncouraged(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's fill property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindFill(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's foreground color property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindForegroundColor(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's geometry property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindGeometry(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's horizontal alignment property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindHorizontalAlignment(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's resref (image) property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindResref(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's label property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindLabel(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's legend property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindLegend(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's line thickness property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindLineThickness(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's max property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindMax(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's min property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindMin(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's placeholder property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindPlaceholder(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's points property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindPoints(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's rectangle property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindRectangle(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's region property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindRegion(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's resizable property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindResizable(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's rowcount property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindRowCount(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's scissor property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindScissor(string sBind, int bWatch = FALSE); + +/// @brief Binds a slider control's bounds. +/// @param sUpper Variable to bind. +/// @param sLower Variable to bind. +/// @param sStep Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindSliderBounds(string sUpper, string sLower, string sStep, int bWatch = FALSE); + +/// @brief Binds the control's step property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindStep(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's text property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindText(string sBind, int bWatch = FALSE); + +/// @brief Binds the form's title property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindTitle(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's tooltip property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +/// @param bDisabledTooltip If TRUE, the control's disabled tooltip property +/// will also be bound to sBind. +void NUI_BindTooltip(string sBind, int bWatch = FALSE, int bDisabledTooltip = FALSE); + +/// @brief Binds the form's elements property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindTransparent(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's type property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindType(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's value property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindValue(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's vertical alignment property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindVerticalAlignment(string sBind, int bWatch = FALSE); + +/// @brief Binds the control's visible property. +/// @param sBind Variable to bind. +/// @param bWatch TRUE to set a bind watch. +void NUI_BindVisible(string sBind, int bWatch = FALSE); + +/// @brief Set the form's accepts input property. +/// @param bAcceptsInput Whether the form will accept player input. +void NUI_SetAcceptsInput(int bAcceptsInput = TRUE); + +/// @brief Sets the control's aspect property. +/// @param nAspect NUI_ASPECT_* constant. +void NUI_SetAspect(int nAspect); + +/// @brief Sets the control's aspect ratio property. +/// @param fAspect Aspect ratio (x/y). +void NUI_SetAspectRatio(float fAspect); + +/// @brief Sets the form's or control's border property. +/// @param bVisible Whether the border is visible. +void NUI_SetBorder(int bVisible = TRUE); + +/// @brief Sets the control's center property. +/// @param x X-coordinate. +/// @param y Y-coordinate. +void NUI_SetCenter(float x, float y); + +/// @brief Sets the form's closable property. +/// @param bClosable Whether the form is closable. +void NUI_SetClosable(int bClosable = TRUE); + +/// @brief Sets the form's collapsible property. +/// @param bCollapsible Whether the form is collapsible. +void NUI_SetCollapsible(int bCollapsible = TRUE); + +/// @brief Binds the control's color property. +/// @param sColor Json-parseable color vector as defined by NUI_DefineColor(), +/// NUI_DefineRGBColor(), NUI_DefineHSVColor(), NUI_DefineHexColor(), and +/// NUI_DefineRandomColor(). +void NUI_SetColor(string sColor); + +/// @brief For bound controls, sets the value that will be initially set +/// to the control's value property. +/// @param sDefault Default value to set. +void NUI_SetDefaultValue(string sDefault); + +/// @brief Sets the control's center property. +/// @param sCenter Json-parseable point vector as defined by NUI_DefinePoint(). +void NUI_SetDefinedCenter(string sCenter); + +/// @brief Sets the form's geometry property. +/// @param sGeometry Json-parseable string representing a rectangular region as +/// defined by NUI_DefineRectangle(). +void NUI_SetDefinedGeometry(string sGeometry); + +/// @brief Sets the control's width and height properties. +/// @param fWidth Width. +/// @param fHeight Height. +void NUI_SetDimensions(float fWidth, float fHeight); + +/// @brief Sets the disabled control's tooltip property. +/// @param sTooltip Tooltip text. +void NUI_SetDisabledTooltip(string sText); + +/// @brief Sets the control's direction property. +/// @param nDirection NUI_ORIENTATION_* constant. +void NUI_SetDirection(int nDirection = NUI_ORIENTATION_ROW); + +/// @brief Sets the control's draw condition property. +/// @param nCondition NUI_CONDITION_* constant. +void NUI_SetDrawCondition(int nCondition = NUI_DRAW_ALWAYS); + +/// @brief Sets the control's draw position property. +/// @param nPosition NUI_POSITION_* constant. +void NUI_SetDrawPosition(int nPosition = NUI_DRAW_ABOVE); + +/// @brief Sets the control's elements property. +/// @param sElements +void NUI_SetElements(string sElements); + +/// @brief Sets the control's enabled property. +/// @param bEnabled Whether the control is enabled. +void NUI_SetEnabled(int bEnabled = TRUE); + +/// @brief Sets the control's encouraged property. +/// @param bEncourage Whether the control is encouraged. +void NUI_SetEncouraged(int bEncouraged = TRUE); + +/// @brief Sets the control's fill property. +/// @param bFill Whether the control is filled. +void NUI_SetFill(int bFill = TRUE); + +/// @brief Sets the bounds for a float-based slider. +/// @param fLower Lower bound. +/// @param fUpper Upper bound. +/// @param fStep Step Value. +void NUI_SetFloatSliderBounds(float fLower, float fUpper, float fStep); + +/// @brief Sets the control's foreground color property. +/// @param sColor A json-parseable color vector as +/// defined by NUI_DefineRGBColor(), NUI_DefineHSVColor(), +/// NUI_DefineHexColor(), or NUI_DefineRandomColor(). +void NUI_SetForegroundColor(string sColor); + +/// @brief Sets the form's geometry property. +/// @param x X-coordinate, top left corner. +/// @param y Y-coordinate, top left corner. +/// @param w Width. +/// @param h Height. +void NUI_SetGeometry(float x, float y, float w, float h); + +/// @brief Sets the control's height property. +/// @param fHeight Height. +void NUI_SetHeight(float fHeight); + +/// @brief Sets the control's horizontal alignment property. +/// @param nAlign NUI_HALIGN_* constant. +void NUI_SetHorizontalAlignment(int nAlign); + +/// @brief Sets the control's id property. +/// @param sID ID. +void NUI_SetID(string sID); + +/// @brief Sets the bounds for a integer-based slider. +/// @param nLower Lower bound. +/// @param nUpper Upper bound. +/// @param nStep Step Value. +void NUI_SetIntSliderBounds(int nLower, int nUpper, int nStep); + +/// @brief Sets the control's label property. +/// @param sLabel Label. +void NUI_SetLabel(string sLabel); + +/// @brief Sets the control's max length property. +/// @param nLength Max number of characters. +void NUI_SetLength(int nLength); + +/// @brief Sets the control's line thickness property. +/// @param fThickness Line thickness. +void NUI_SetLineThickness(float fThickness); + +/// @brief Sets the control's margin property. +/// @param fMargin Margin. +void NUI_SetMargin(float fMargin); + +/// @brief Sets the control's multiline property. +/// @param bMultiline Whether the textbox has multiple lines. +void NUI_SetMultiline(int bMultiline = TRUE); + +/// @brief Sets the control's padding property. +/// @param fPadding Padding. +void NUI_SetPadding(float fPadding); + +/// @brief Sets the control's placeholder property. +/// @param sText Placeholder text. +void NUI_SetPlaceholder(string sText); + +/// @brief Sets the control's points property. +/// @param sPoint Json-parseable string representing a coordinate array as defined +/// by NUI_GetLinePoints(), NUI_AddLinePoint(), or NUI_GetRectanglePoints(). +void NUI_SetPoints(string sPoints); + +/// @brief Sets the control's radius property. +/// @param r Radius. +void NUI_SetRadius(float r); + +/// @brief Sets the control's rectangle property. +/// @param sRectangle Json-parsaable string representing a rectangular region as defined +/// by NUI_DefineRectangle(). +void NUI_SetRectangle(string sRectangle); + +/// @brief Sets the control's region property. +/// @param sRegion Json-parsaable string representing a rectangular region as defined +/// by NUI_DefineRectangle(). +void NUI_SetRegion(string sRegion); + +/// @brief Sets the form's resizable property. +/// @param bResizable Whether the form is resizable. +void NUI_SetResizable(int bResizable = TRUE); + +/// @brief Sets the control's resref/image property. +/// @param sResref Resource resref. +void NUI_SetResref(string sResref); + +/// @brief Sets the control's row count property. +/// @param nRowCount Number of rows. +void NUI_SetRowCount(int nRowCount); + +/// @brief Sets the control's row height property. +/// @param fRowHeight Row height. +void NUI_SetRowHeight(float fRowHeight); + +/// @brief Sets the control's scissor property. +/// @param bScissor Whether to scissor to the control's dimensions. +void NUI_SetScissor(int bScissor); + +/// @brief Sets the control's scrollbars property. +/// @param nScrollBars NUI_SCROLLBARS_* constant. +void NUI_SetScrollbars(int nScrollbars = NUI_SCROLLBARS_AUTO); + +/// @brief Sets the control's width and height properties. +/// @param fSide Length of one side. +void NUI_SetSquare(float fSide); + +/// @brief Sets the control's type property. +/// @note Sets an editable textbox to non-editable. +void NUI_SetStatic(); + +/// @brief Sets the template's variable property. +/// @param bVariable Whether the template control width is variable. +void NUI_SetTemplateVariable(int bVariable = TRUE); + +/// @brief Sets the template's width property. +/// @param fWidth Width. +void NUI_SetTemplateWidth(float fWidth); + +/// @brief Sets the control's text property. +/// @param sText Text. +void NUI_SetText(string sText); + +/// @brief Sets the form's title property. +/// @param sTitle Title. +void NUI_SetTitle(string sTitle); + +/// @brief Sets the control's tooltip property. +/// @param sText Tooltip text. +/// @param bDisabledTooltip If TRUE, the control's disabled tooltip property +/// will also be set to sText. +void NUI_SetTooltip(string sText, int bDisabledTooltip = FALSE); + +/// @brief Sets the form's transparent property. +/// @param bTransparent Whether the form's background is transparent. +void NUI_SetTransparent(int bTransparent = TRUE); + +/// @brief Sets the control's value property. +/// @param sValue Json-Parseable string. +/// @note sValue must be a string representing a value json structure. +void NUI_SetValue(string sValue); + +/// @brief Sets the control's vertical alignment property. +/// @param nAlign NUI_VALIGN_* constant. +void NUI_SetVerticalAlignment(int nAlign); + +/// @brief Sets the control's visible property. +/// @param bVisible Whether the control is visible. +void NUI_SetVisible(int bVisible = TRUE); + +/// @brief Sets the control's width property. +/// @param fWidth Width. +void NUI_SetWidth(float fWidth); + +/// @brief Set's the control's wordwrap property. +/// @param bWrap Whether the text wrap's withing the control's width. +void NUI_SetWordWrap(int bWrap = TRUE); + +/// @brief Defines all forms via formfiles that match prefixes set into +/// the configuration file. +/// @param sFormfile Optional formfile specification. +/// @note If sFormfile is passed, only the specified formfile will be loaded. +void NUI_DefineForms(string sFormfile = ""); + +/// @brief Display an NUI form. +/// @param oPC Client to display the form on. +/// @param sFormID ID of the for to display. +/// @param sProfile Optional form profile. +/// @returns Form's token as assigned by the game engine. +int NUI_DisplayForm(object oPC, string sFormID, string sProfile = "default"); + +/// @brief Close an open NUI form. +/// @param oPC Client on which to close the form. +/// @param sFormID ID of the form to close. +void NUI_CloseForm(object oPC, string sFormID); + +/// @brief Display a subform onto control group element or form root. +/// @param oPC Client to display the subform on. +/// @param sFormID Form ID. +/// @param sElementID Element ID to replace. +/// @param sSubformID Subform ID to insert into sElement. +void NUI_DisplaySubform(object oPC, string sFormID, string sElementID, string sSubformID); + +/// @brief Creates the default profile. +void NUI_CreateDefaultProfile(); + +/// @brief Create a custom form profile based on the default profile. +/// @param sProfile Profile name. +/// @param sBase Optional; must be a previously created profile. If set, +/// sProfile will be based on profile sBase. +void NUI_CreateProfile(string sProfile, string sBase = ""); + +/// @brief Set a profile default bind value. +/// @param sBind Bind name. +/// @param sJson Json-parseable string representing the bind's profile value. +void NUI_SetProfileBind(string sBind, string sJson); + +/// @brief Set a profile default bind value. +/// @param sBind Bind name. +/// @param jValue Json bind value. +void NUI_SetProfileBindJ(string sBind, json jValue); + +/// @brief Set multiple profile binds to a single value. +/// @param sBinds Comma-delimited list of bind names. +/// @param sJson Json-parseable string representing the binds' profile value. +void NUI_SetProfileBinds(string sBinds, string sJson); + +/// @brief Set a form's profile. +/// @param oPC Client to set the profile on. +/// @param sFormID Form ID to set the profile on. +/// @param sProfile Profile to set. +void NUI_SetProfile(object oPC, string sFormID, string sProfile); + +/// @brief Set a bind value. +/// @param oPC Player to set the bind for. +/// @param sFormID Form ID. +/// @param sBind Bind/property name. +/// @param sValue Json-parseable bind Value. +void NUI_SetBind(object oPC, string sFormID, string sBind, string sValue); + +/// @brief Set a bind value. +/// @param oPC Player to set the bind for. +/// @param sFormID Form ID. +/// @param sBind Bind/property name. +/// @param jValue Json bind value. +void NUI_SetBindJ(object oPC, string sFormID, string sBind, json jValue); + +/// @brief Set a bind watch. +/// @param oPC Player to set the bind for. +/// @param sFormID Form ID. +/// @param sBind Bind/property name. +/// @param bWatch TRUE to set the watch, FALSE to remove it. +void NUI_SetBindWatch(object oPC, string sFormID, string sBind, int bWatch = TRUE); + +/// @brief Retireve a bind value. +/// @param oPC Player to set the bind for. +/// @param sFormID Form ID. +/// @param sBind Bind/property name. +/// @returns Current bind value. +json NUI_GetBind(object oPC, string sFormID, string sBind); + +/// @brief Create a temporary layout to be immediately returned by NUI_GetLayout(). +/// @note Allow the creation of temporary layouts that can be immediately inserted +/// into a form's layout. This function should not be used for creating +/// tabs to be referenced at a later time. The layout created with this function +/// is temporary. +void NUI_CreateLayout(); + +/// @brief Get the temporary layout created with NUI_CreateLayout(). +/// @note The json-parseable string can be inserted directly into the current layout +/// or used in a layout replacement (tabs). This data will not, however, be +/// saved to the database, so this function and NUI_CreateLayout(). are +/// meant for dynamic form building. +string NUI_GetLayout(); + +/// @brief Determines if a form is currently open. +/// @param oPC Players to check for the form. +/// @param sFormID The form's ID. +/// @returns TRUE if oPC has sFormID open, FALSE otherwise. +int NUI_GetIsFormOpen(object oPC, string sFormID); + +/// @brief Save all of sFormID's bind values to a local variable. +/// @param oPC Player associated with sFormID. +/// @param sFormID Form ID to save bind values for. +void NUI_SaveBindState(object oPC, string sFormID); + +/// @brief Restore all of sFormID's bind values from a local variable. +/// @param oPC Player associated with sFormID. +/// @param sFormID Form ID to restore bind values for. +void NUI_RestoreBindState(object oPC, string sFormID); + +/// @brief Returns all events data as a struct NUIEventData. +/// @note Only valid within HandleNUIEvents() event handler. +struct NUIEventData NUI_GetEventData(); + +/// @note Temporary prototypes to allow .35 function to work when .35 is not installed. +/// TODO Remove when .35 is stable. +//json RegExpMatch(string sRegExp, string sValue, int nSyntaxFlags = REGEXP_ECMASCRIPT, int nMatchFlags = REGEXP_FORMAT_DEFAULT); +//string RegExpReplace(string sRegExp, string sValue, string sReplacement, int nSyntaxFlags = REGEXP_ECMASCRIPT, int nMatchFlags = REGEXP_FORMAT_DEFAULT); + +// ----------------------------------------------------------------------------- +// Public Functions +// Administrative Helpers +// ----------------------------------------------------------------------------- + +// TODO Need prototypes +string NUI_GetKey(string sPair) +{ + int nIndex; + if ((nIndex = FindSubString(sPair, ":")) == -1) + nIndex = FindSubString(sPair, "="); + + if (nIndex == -1) return sPair; + else return GetSubString(sPair, 0, nIndex); +} + +string NUI_GetValue(string sPair) +{ + int nIndex; + if ((nIndex = FindSubString(sPair, ":")) == -1) + nIndex = FindSubString(sPair, "="); + + if (nIndex == -1) return ""; + else return GetSubString(sPair, ++nIndex, GetStringLength(sPair)); +} + +object nui_GetDataObject() {return GetModule();} + +// ----------------------------------------------------------------------------- +// Private Functions +// Build Flags and JSON Pathing +// ----------------------------------------------------------------------------- + +// TODO +/// CUT LINE FOR .35 - Remove below once .35 is stable + +string nui_ReplaceSubString(string sString, string sSub, int nStart, int nEnd) +{ + int nLength = GetStringLength(sString); + if (nStart < 0 || nStart >= nLength || nStart > nEnd) + return sString; + + return GetSubString(sString, 0, nStart) + sSub + + GetSubString(sString, nEnd + 1, nLength - nEnd); +} + +string nui_SubstituteSubString(string sString, string sToken, string sSub) +{ + int nPos; + if ((nPos = FindSubString(sString, sToken)) == -1) + return sString; + + return nui_ReplaceSubString(sString, sSub, nPos, nPos + GetStringLength(sToken) - 1); +} + +string nui_SubstituteSubStrings(string sString, string sToken, string sSub) +{ + int n; + while ((n = FindSubString(sString, sToken)) >= 0) + sString = nui_SubstituteSubString(sString, sToken, sSub); + + return sString; +} + +int nui_GetMatchesPattern(string sString, string sPattern) +{ + sqlquery q = SqlPrepareQueryObject(GetModule(), + "SELECT @string GLOB @pattern;"); + SqlBindString(q, "@string", sString); + SqlBindString(q, "@pattern", sPattern); + return SqlStep(q) ? SqlGetInt(q, 0) : FALSE; +} + +string nui_RegExpReplace(string sToken, string sString, string sSub) +{ + return nui_SubstituteSubStrings(sString, sToken, sSub); +} + +int nui_RegExpMatch(string sString) +{ + return nui_GetMatchesPattern(sString, "*row_template???") || nui_GetMatchesPattern(sString, "*row_template?????"); +} + +string nui_RegExpReplaceLast(string sToken, string sString, string sSub) +{ + int nCount = GetSubStringCount(sString, sToken); + + if (nCount) + { + int nPos = FindSubStringN(sString, sToken, nCount - 1); + + // Need to delete the rest of the path too! + return nui_ReplaceSubString(sString, sSub, nPos, nPos + GetStringLength(sString)); + //return nui_ReplaceSubString(sString, sSub, nPos, nPos + GetStringLength(sToken) - 1); + } + else + return sString; +} + +/// CUT LINE FOR .35 - Remove above once .35 is stable + +void nui_ClearVariables() +{ + object oData = nui_GetDataObject(); + DeleteLocalInt (oData, "FLAG_DEFINITION"); + DeleteLocalInt (oData, "FLAG_INCREMENT"); + DeleteLocalInt (oData, "FLAG_DRAWLIST"); + DeleteLocalInt (oData, "FLAG_LISTBOX"); + DeleteLocalInt (oData, "FLAG_FORCE"); + DeleteLocalInt (oData, "NUI_ENTRY_COUNT"); + DeleteLocalString(oData, "NUI_CONTROL_TYPE"); + DeleteLocalString(oData, "NUI_PATH"); + DeleteLocalString(oData, "NUI_FORMID"); +} + +int nui_ToggleFlag(int nFlag, string sFlag) +{ + if (nFlag == -1) + nFlag = !GetLocalInt(nui_GetDataObject(), sFlag); + + SetLocalInt(nui_GetDataObject(), sFlag, nFlag); + return nFlag; +} + +int nui_GetEntryCount() {return GetLocalInt(nui_GetDataObject(), "NUI_ENTRY_COUNT");} +void nui_ResetEntryCount() {DeleteLocalInt(nui_GetDataObject(), "NUI_ENTRY_COUNT");} + +int nui_IncrementEntryCount(int nIncrement = 1) +{ + int nCount = nui_GetEntryCount(); + SetLocalInt(nui_GetDataObject(), "NUI_ENTRY_COUNT", ++nCount); + return nCount; +} + +int nui_ToggleIncrementFlag(int nFlag = -1) {return nui_ToggleFlag(nFlag, "FLAG_INCREMENT");} +int nui_GetIncrementFlag() {return GetLocalInt(nui_GetDataObject(), "FLAG_INCREMENT");} + +int nui_ToggleDrawlistFlag(int nFlag = -1) {return nui_ToggleFlag(nFlag, "FLAG_DRAWLIST");} +int nui_GetDrawlistFlag() {return GetLocalInt(nui_GetDataObject(), "FLAG_DRAWLIST");} + +int nui_ToggleListboxFlag(int nFlag = -1) {return nui_ToggleFlag(nFlag, "FLAG_LISTBOX");} +int nui_GetListboxFlag() {return GetLocalInt(nui_GetDataObject(), "FLAG_LISTBOX");} + +int nui_ToggleDefinitionFlag(int nFlag = -1) {return nui_ToggleFlag(nFlag, "FLAG_DEFINITION");} +int nui_GetDefinitionFlag() {return GetLocalInt(nui_GetDataObject(), "FLAG_DEFINITION");} + +void nui_SetControlType(string sType) {SetLocalString(nui_GetDataObject(), "NUI_CONTROL_TYPE", sType);} +string nui_GetControlType() {return GetLocalString(nui_GetDataObject(), "NUI_CONTROL_TYPE");} + +string nui_SetPath(string sPath) +{ + SetLocalString(nui_GetDataObject(), "NUI_PATH", sPath); + return sPath; +} + +void nui_ResetPath() {nui_SetPath("$");} +string nui_GetPath() {return GetLocalString(nui_GetDataObject(), "NUI_PATH");} + +void nui_SetFormID(string sFormID) {SetLocalString(nui_GetDataObject(), "NUI_FORMID", sFormID);} +string nui_GetFormID() {return GetLocalString(nui_GetDataObject(), "NUI_FORMID");} + +void nui_SetFormfile(string sFormfile) {SetLocalString(nui_GetDataObject(), "NUI_FORMFILE", sFormfile);} +string nui_GetFormfile() {return GetLocalString(nui_GetDataObject(), "NUI_FORMFILE");} + +// TODO When .35 is stable, remove the `nui_` in front of both `nui_RegExpReplace` functions +// to restore .35 functionality. +string nui_SubstitutePath(string sSub) {return nui_SetPath(nui_RegExpReplace("@", nui_GetPath(), sSub));} +string nui_GetSubstitutedPath(string sSub) {return nui_RegExpReplace("@", nui_GetPath(), sSub);} + +// TODO Restore `RegExpMatch` when .35 is stable. +string nui_GetGroupKey() +{ + //return JsonGetString(JsonArrayGet(RegExpMatch("\\.(\\w*)\\[(?!.*\\.\\w*\\[)", nui_GetPath()), 1)); + return nui_RegExpMatch(nui_GetPath()) ? "row_template" : ""; +} + +string nui_IncrementPath(string sElement = "", int bForce = FALSE) +{ + if (!nui_GetIncrementFlag() && !bForce) + return nui_GetPath(); + else + nui_ToggleIncrementFlag(); + + string sPath = nui_GetPath(); + + if (sPath == "$") + sPath += ".root"; + else + { + sPath = nui_SubstitutePath("#-1"); + + if (nui_GetGroupKey() == "row_template" && (nui_GetControlType() == "group" || sElement == "draw_list")) + sPath += "[0]"; + + if (sElement != "draw_list") + { + if (nui_GetControlType() == "listbox") + sPath += ".row_template[@]"; + else + sPath += ".children[@]"; + } + else + sPath += ".draw_list[@]"; + } + + return nui_SetPath(sPath); +} + +// TODO Restore .35 functionality + +string nui_DecrementPath(int n = 1) +{ + string sPath; + while (n-- > 0) + sPath = nui_SetPath(nui_RegExpReplaceLast("[#-1]", nui_GetPath(), "[@]")); + //sPath = nui_SetPath(RegExpReplace("\\[#-1\\](?!.*\\[#-1\\]).*", nui_GetPath(), "[@]")); + + return sPath; +} + +// ----------------------------------------------------------------------------- +// Private Functions +// Database +// ----------------------------------------------------------------------------- + +string sQuery; +sqlquery sql; + +void nui_BeginTransaction() {SqlStep(SqlPrepareQueryObject(GetModule(), "BEGIN TRANSACTION;"));} +void nui_CommitTransaction() {SqlStep(SqlPrepareQueryObject(GetModule(), "COMMIT TRANSACTION;"));} + +sqlquery nui_PrepareQuery(string sQuery, int bForceModule = FALSE) +{ + if (nui_GetDefinitionFlag() || bForceModule) + return SqlPrepareQueryObject(GetModule(), sQuery); + + if (NUI_USE_CAMPAIGN_DATABASE) + return SqlPrepareQueryCampaign(NUI_DATABASE, sQuery); + else + return SqlPrepareQueryObject(GetModule(), sQuery); +} + +void nui_InitializeDatabase() +{ + sQuery = "DROP TABLE IF EXISTS nui_forms;"; + SqlStep(nui_PrepareQuery(sQuery)); + SqlStep(nui_PrepareQuery(sQuery, TRUE)); + + sQuery = "CREATE TABLE IF NOT EXISTS nui_forms (" + + "id INTEGER PRIMARY KEY AUTOINCREMENT, " + + "form TEXT NOT NULL UNIQUE, " + + "definition BLOB);"; + SqlStep(nui_PrepareQuery(sQuery)); + SqlStep(nui_PrepareQuery(sQuery, TRUE)); + + sQuery = "DROP TABLE IF EXISTS nui_fi_data;"; + SqlStep(nui_PrepareQuery(sQuery)); + + sQuery = + "CREATE TABLE IF NOT EXISTS nui_fi_data (" + + "event_id INTEGER PRIMARY KEY AUTOINCREMENT, " + + "sPlayer INTEGER, " + + "sFormID STRING, " + + "nToken INTEGER, " + + "sEvent STRING, " + + "sControlID STRING, " + + "nIndex INTEGER, " + + "jPayload STRING, " + + "binds_before STRING, " + + "binds_after STRING, " + + "timestamp INTEGER);"; + SqlStep(nui_PrepareQuery(sQuery)); +} + +void nui_SaveForm(string sID, string sJson) +{ + sQuery = "INSERT INTO nui_forms (form, definition) " + + "VALUES (@form, json(@json)) " + + "ON CONFLICT (form) DO UPDATE SET " + + "definition = json(@json);"; + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sID); + SqlBindString(sql, "@json", sJson); + SqlStep(sql); +} + +void nui_DeleteForm(string sID) +{ + sql = nui_PrepareQuery("DELETE FROM nui_forms WHERE form = @form;"); + SqlBindString(sql, "@form", sID); + SqlStep(sql); +} + +void nui_CopyDefinitions(string sTable = "nui_forms") +{ + if (!NUI_USE_CAMPAIGN_DATABASE) + return; + + sQuery = "WITH forms AS (SELECT json_object('form', form, 'definition', definition) AS f " + + "FROM " + sTable + ") SELECT json_group_array(json(f)) FROM forms;"; + sql = nui_PrepareQuery(sQuery, TRUE); + json jForms = SqlStep(sql) ? SqlGetJson(sql, 0) : JsonNull(); + + if (jForms == JsonNull()) + return; + + // ->> only works in .35! D'oh! + /* + sQuery = "INSERT OR REPLACE INTO " + sTable + " (form, definition) " + + "SELECT value ->> '$.form', value ->> '$.definition' " + + "FROM (SELECT value FROM json_each(@forms));"; + */ + // TODO remove when .35 is live + sQuery = "INSERT OR REPLACE INTO " + sTable + " (form, definition) " + + "SELECT json_extract(value, '$.form'), json_extract(value, '$.definition') " + + "FROM (SELECT value FROM json_each(@forms));"; + sql = nui_PrepareQuery(sQuery); + SqlBindJson(sql, "@forms", jForms); + SqlStep(sql); + + sQuery = "DELETE FROM " + sTable + ";"; + SqlStep(nui_PrepareQuery(sQuery, TRUE)); +} + +string nui_GetDefinitionValue(string sFormID, string sPath = "") +{ + sQuery = "SELECT json_extract(definition, '$" + (sPath == "" ? "" : "." + sPath) + "') " + + "FROM nui_forms WHERE form = @form;"; + + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sFormID); + return SqlStep(sql) ? SqlGetString(sql, 0) : ""; +} + +// ----------------------------------------------------------------------------- +// Private Functions +// Form Definition +// ----------------------------------------------------------------------------- + +string nui_CreateRowTemplate(string sControl) +{ + return "[" + sControl + "," + nuiFloat(25.0) + "," + nuiBool(TRUE) + "]"; +} + +void nui_SetObject(string sProperty, string sValue, string sType = "") +{ + string sPath, sFormID = nui_GetFormID(); + + if (sProperty != "") + { + sPath = nui_GetSubstitutedPath("#-1"); + + if (nui_GetGroupKey() == "row_template") + { + if (sProperty == "NUI_TEMPLATE_WIDTH") sPath += "[1]"; + else if (sProperty == "NUI_TEMPLATE_VARIABLE") sPath += "[2]"; + else sPath += "[0]." + sProperty; + } + else if (sProperty == "NUI_ELEMENT") sPath += ".elements[#]"; + else if (sProperty == "NUI_SERIES") sPath += ".value[#]"; + else sPath += "." + sProperty; + } + else + { + nui_IncrementPath(sType); + + if (nui_GetGroupKey() == "row_template") + sValue = nui_CreateRowTemplate(sValue); + + if (sType != "") + { + if (sType == "combo" || sType == "options" || sType == "tabbar") + nui_ResetEntryCount(); + + nui_SetControlType(sType); + } + + sPath = nui_GetSubstitutedPath("#"); + } + + sQuery = "UPDATE nui_forms SET definition = (SELECT json_set(definition, '" + sPath + + "', json(@value)) FROM nui_forms WHERE form = @form) WHERE form = @form;"; + + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sFormID); + SqlBindString(sql, "@value", sValue); + SqlStep(sql); +} + +void nui_SetControl (string sValue, string sType = "") {nui_SetObject("", sValue, sType);} +void nui_SetProperty(string sProperty, string sValue) {nui_SetObject(sProperty, sValue);} + +void nui_CreateControl(string sType, string sID = "") +{ + string sControl = "{" + + nuiString("label") + ":null," + + nuiString("value") + ":null," + + nuiString("enabled") + ":true," + + nuiString("visible") + ":true," + + nuiString("user_data") + ":{}," + + nuiString("event_data") + ":{}," + + nuiString("type") + ":" + nuiString(sType) + + (sID == "" ? "" : "," + + nuiString("id") + ":" + nuiString(sID)) + + "}"; + + nui_SetControl(sControl, sType); +} + +string nui_CreateCanvasTemplate(string sProperties) +{ + return "{" + + nuiString("enabled") + ":true," + + nuiString("fill") + ":false," + + nuiString("line_thickness") + ":0.5," + + nuiString("order") + ":" + nuiInt(NUI_DRAW_ABOVE) + "," + + nuiString("render") + ":" + nuiInt(NUI_DRAW_ALWAYS) + "," + + nuiString("color") + ":" + NUI_DefineRGBColor(255, 255, 255) + "," + + sProperties + + "}"; +} + +// ----------------------------------------------------------------------------- +// Public Functions +// ----------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Json-Parseable Creators +// ----------------------------------------------------------------------------- + +string nuiString(string s) {return "\"" + s + "\"";} +string nuiInt(int n) {return IntToString(n);} +string nuiFloat(float f) {return FloatToString(f);} +string nuiBool(int b) {return b ? "true" : "false";} +string nuiStrRef(int nStrRef) {return "{" + nuiString("strref") + ":" + nuiInt(nStrRef) + "}";} +string nuiNull() {return "null";} +int nuiIsBind(string s) {return JsonGetType(JsonParse(s)) == JSON_TYPE_OBJECT;} + +string nuiBind(string sBind, int bWatch = FALSE) +{ + return "{" + nuiString("bind") + ":" + nuiString(sBind) + "," + + nuiString("watch") + ":" + nuiBool(bWatch) + "}"; +} + +// ----------------------------------------------------------------------------- +// Form/Subform/Layout Definition +// ----------------------------------------------------------------------------- + +void NUI_Initialize() +{ + if (GetLocalInt(GetModule(), "NUI_INITIALIZED")) + return; + + SetLocalInt(GetModule(), "NUI_INITIALIZED", TRUE); + + nui_InitializeDatabase(); + NUI_DefineForms(); +} + +void NUI_CreateForm(string sID, string sVersion = "") +{ + nui_SetFormID(sID); + + string sForm = "{" + + nuiString("accepts_input") + ":true," + + nuiString("border") + ":true," + + nuiString("closable") + ":true," + + nuiString("resizable") + ":true," + + nuiString("collapsed") + ":false," + + nuiString("transparent") + ":false," + + nuiString("version") + ":1," + + nuiString("user_data") + ":{}," + + nuiString("event_data") + ":[]," + + nuiString("subforms") + ":{}," + + nuiString("profiles") + ":{" + nuiString("default") + ":{}}," + + nuiString("formfile") + ":" + nuiString(nui_GetFormfile()) + "," + + nuiString("geometry") + ":" + nuiBind("geometry", TRUE) + "," + + nuiString("title") + ":" + nuiBind("title") + "," + + nuiString("local_version") + ":" + nuiString(sVersion) + "," + + nuiString("id") + ":" + nuiString(sID) + + "}"; + + nui_SaveForm(sID, sForm); + nui_ResetPath(); +} + +void NUI_CreateSubform(string sID) {nui_SetPath("$.subforms." + sID);} + +void NUI_CreateLayout() +{ + nui_ToggleDefinitionFlag(TRUE); + NUI_CreateForm("__layout__"); +} + +string NUI_GetLayout() +{ + string sLayout = nui_GetDefinitionValue("__layout__", "root"); + nui_DeleteForm("__layout__"); + nui_ToggleDefinitionFlag(FALSE); + return sLayout; +} + +json NUI_GetBindState(object oPC, string sFormID) +{ + int n, nToken = NuiFindWindow(oPC, sFormID); + json jState = JsonObject(); + string sBind; + + if (nToken) + { + while ((sBind = NuiGetNthBind(oPC, nToken, FALSE, n++)) != "") + jState = JsonObjectSet(jState, sBind, NuiGetBind(oPC, nToken, sBind)); + } + + return jState; +} + +void NUI_SaveBindState(object oPC, string sFormID) +{ + json jState = NUI_GetBindState(oPC, sFormID); + + if (jState != JsonObject()) + SetLocalJson(oPC, "NUI_STATE:" + sFormID, jState); +} + +void NUI_RestoreBindState(object oPC, string sFormID) +{ + json jState = GetLocalJson(oPC, "NUI_STATE:" + sFormID); + if (jState == JsonNull()) + return; + + json jKeys = JsonObjectKeys(jState); + int n, nToken = NuiFindWindow(oPC, sFormID); + string sBind; + + while ((sBind = JsonGetString(JsonArrayGet(jKeys, n++))) != "") + NuiSetBind(oPC, nToken, sBind, JsonObjectGet(jState, sBind)); +} + +// ----------------------------------------------------------------------------- +// Vectors +// ----------------------------------------------------------------------------- + +string NUI_DefinePoint(float x, float y) +{ + return "{" + + nuiString("x") + ":" + nuiFloat(x) + "," + + nuiString("y") + ":" + nuiFloat(y) + + "}"; +} + +string NUI_GetLinePoints(float x1, float y1, float x2, float y2) +{ + return "[" + + nuiFloat(x1) + "," + + nuiFloat(y1) + "," + + nuiFloat(x2) + "," + + nuiFloat(y2) + + "]"; +} + +string NUI_AddLinePoint(string sPoints, float x, float y) +{ + string sOpen; + if (sPoints == "" || sPoints == "[]") + sOpen = "["; + else + sOpen = GetStringLeft(sPoints, GetStringLength(sPoints) - 1) + ","; + + return sOpen + + nuiFloat(x) + "," + + nuiFloat(y) + "]"; +} + +string NUI_DefineRGBColor(int r, int g, int b, int a = 255) +{ + return "{" + + nuiString("r") + ":" + nuiInt(r) + "," + + nuiString("g") + ":" + nuiInt(g) + "," + + nuiString("b") + ":" + nuiInt(b) + "," + + nuiString("a") + ":" + nuiInt(a) + + "}"; +} + +string NUI_DefineHSVColor(float h, float s, float v) +{ + struct HSV hsv; + hsv.h = h; + hsv.s = s; + hsv.v = v; + + struct RGB rgb = HSVToRGB(hsv); + return NUI_DefineRGBColor(rgb.r, rgb.g, rgb.b); +} + +string NUI_DefineHexColor(int nColor) +{ + struct RGB rgb = HexToRGB(nColor); + return NUI_DefineRGBColor(rgb.r, rgb.g, rgb.b); +} + +string NUI_DefineRandomColor() +{ + return NUI_DefineRGBColor(Random(256), Random(256), Random(256)); +} + +string NUI_DefineRectangle(float x, float y, float w, float h) +{ + return "{" + + nuiString("x") + ":" + nuiFloat(x) + "," + + nuiString("y") + ":" + nuiFloat(y) + "," + + nuiString("w") + ":" + nuiFloat(w) + "," + + nuiString("h") + ":" + nuiFloat(h) + + "}"; +} + +string NUI_GetRectanglePoints(float x, float y, float w, float h) +{ + string sx = nuiFloat(x); + string sy = nuiFloat(y); + string sxw = nuiFloat(x + w); + string syh = nuiFloat(y + h); + + return "[" + + sx + "," + sy + "," + + sxw + "," + sy + "," + + sxw + "," + syh + "," + + sx + "," + syh + "," + + sx + "," + sy + + "]"; +} + +// TODO Needs Testing +string NUI_GetDefinedRectanglePoints(string sRectangle) +{ + float x = StringToFloat(NUI_GetValue(GetListItem(sRectangle, 0))); + float y = StringToFloat(NUI_GetValue(GetListItem(sRectangle, 1))); + float w = StringToFloat(NUI_GetValue(GetListItem(sRectangle, 2))); + float h = StringToFloat(NUI_GetValue(GetListItem(sRectangle, 3))); + + return NUI_GetRectanglePoints(x, y, w, h); +} + +string NUI_DefineCircle(float x, float y, float r) +{ + r = fclamp(r, 0.0, fmin(x, y)); + string d = nuiFloat(2 * r); + + return "{" + + nuiString("x") + ":" + nuiFloat(x - r) + "," + + nuiString("y") + ":" + nuiFloat(y - r) + "," + + nuiString("w") + ":" + d + "," + + nuiString("h") + ":" + d + + "}"; +} + +// ----------------------------------------------------------------------------- +// Controls +// ----------------------------------------------------------------------------- + +// Columns and Rows ------------------------------------------------------------ + +void nui_AddLayout(string sType, float f = -1.0) +{ + string sDimension; + if (f > 0.0) + sDimension = "," + nuiString(sType == "col" ? "width" : "height") + ":" + nuiFloat(f); + + string sRoot = "{" + + nuiString("enabled") + ":true," + + nuiString("visible") + ":true," + + nuiString("children") + ":[]," + + nuiString("type") + ":" + nuiString(sType) + + sDimension + + "}"; + + if (nui_GetPath() == "$") + { + nui_SetProperty("root", sRoot); + nui_IncrementPath("", TRUE); + } + else + { + nui_SetControl(sRoot, sType); + nui_ToggleIncrementFlag(TRUE); + } +} + +void NUI_AddColumn(float fWidth = -1.0) {nui_AddLayout("col", fWidth);} +void NUI_CloseColumn() {nui_DecrementPath();} + +void NUI_AddRow(float fHeight = -1.0) {nui_AddLayout("row", fHeight);} +void NUI_CloseRow() {nui_DecrementPath();} + +// Controls -------------------------------------------------------------------- + +void NUI_AddCheckbox(string sID = "") {nui_CreateControl("check", sID);} +void NUI_AddColorPicker(string sID = "") {nui_CreateControl("color_picker", sID);} +void NUI_AddCommandButton(string sID = "") {nui_CreateControl("button", sID);} +void NUI_AddFloatSlider(string sID = "") {nui_CreateControl("sliderf", sID);} +void NUI_AddIntSpinBar(string sID = "") {nui_CreateControl("spinbar", sID);} +void NUI_AddImage(string sID = "") {nui_CreateControl("image", sID);} +void NUI_AddImageButton(string sID = "") {nui_CreateControl("button_image", sID);} +void NUI_AddIntSlider(string sID = "") {nui_CreateControl("slider", sID);} +void NUI_AddLabel(string sID = "") {nui_CreateControl("label", sID);} +void NUI_AddMoviePlayer(string sID = "") {nui_CreateControl("movieplayer", sID);} +void NUI_AddProgressBar(string sID = "") {nui_CreateControl("progress", sID);} +void NUI_AddSpacer(string sID = "") {nui_CreateControl("spacer", sID);} +void NUI_AddToggleButton(string sID = "") {nui_CreateControl("button_select", sID);} + +void NUI_AddCanvas() +{ + nui_SetProperty("draw_list", "[]"); + nui_SetProperty("draw_list_scissor", nuiBool(FALSE)); + nui_IncrementPath("draw_list", TRUE); + nui_ToggleDrawlistFlag(TRUE); +} + +void NUI_AddChart(string sID = "") +{ + nui_CreateControl("chart", sID); + nui_SetProperty("value", "[]"); +} + +void NUI_CloseCanvas() +{ + nui_ToggleDrawlistFlag(FALSE); + nui_DecrementPath(); +} + +void NUI_AddCombobox(string sID = "") +{ + nui_CreateControl("combo", sID); + nui_SetProperty("elements", "[]"); +} + +void NUI_AddGroup(string sID = "") +{ + sID = (sID == "" ? "" : nuiString("id") + ":" + nuiString(sID) + ","); + string sGroup = "{" + sID + + nuiString("children") + ":[]," + + nuiString("border") + ":true," + + nuiString("type") + ":" + nuiString("group") + "," + + nuiString("scrollbars") + ":" + nuiInt(NUI_SCROLLBARS_AUTO) + + "}"; + + nui_SetControl(sGroup, "group"); + nui_ToggleIncrementFlag(TRUE); +} + +void NUI_CloseGroup() {nui_DecrementPath();} + +void NUI_AddListbox(string sID = "") +{ + sID = (sID == "" ? "" : nuiString("id") + ":" + nuiString(sID) + ","); + string sList = "{" + sID + + nuiString("row_template") + ":[]," + + nuiString("row_count") + ":1," + + nuiString("border") + ":true," + + nuiString("row_height") + ":25.0," + + nuiString("type") + ":" + nuiString("list") + "," + + nuiString("scrollbars") + ":" + nuiInt(NUI_SCROLLBARS_Y) + + "}"; + + nui_SetControl(sList, "listbox"); + nui_ToggleIncrementFlag(TRUE); +} + +void NUI_CloseListbox() {nui_DecrementPath();} + +void NUI_AddOptionGroup(string sID = "") +{ + nui_CreateControl("options", sID); + nui_SetProperty("elements", "[]"); +} + +// TODO prototype +void NUI_AddToggleGroup(string sID = "") +{ + nui_CreateControl("tabbar", sID); + nui_SetProperty("elements", "[]"); +} + +void NUI_AddTextbox(string sID = "") +{ + nui_CreateControl("textedit", sID); + nui_SetProperty("wordwrap", nuiBool(TRUE)); +} + +void NUI_AddTest(string sID = "") +{ + nui_CreateControl("menutest", sID); +} + +void NUI_SetTest(int n) +{ + //nui_SetProperty("label", nuiString(sLabel)); + //nui_SetProperty("image", nuiString(sImage)); + nui_SetProperty("image_rotation", nuiInt(n)); +} + +// Drawlist -------------------------------------------------------------------- + +void NUI_DrawLine(string sPoints) +{ + string sDraw = + nuiString("points") + ":" + sPoints + "," + + nuiString("type") + ":" + nuiInt(0); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawRectangle(float x, float y, float w, float h) +{ + NUI_DrawLine(NUI_GetRectanglePoints(x, y, w, h)); +} + +void NUI_DrawDefinedRectangle(string sRect) +{ + NUI_DrawLine(NUI_GetDefinedRectanglePoints(sRect)); +} + +void NUI_DrawCircle(float x, float y, float r) +{ + string sDraw = + nuiString("rect") + ":" + NUI_DefineCircle(x, y, r) + "," + + nuiString("type") + ":" + nuiInt(2); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawDefinedCircle(string sCircle) +{ + string sDraw = + nuiString("rect") + ":" + sCircle + "," + + nuiString("type") + ":" + nuiInt(2); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawTextbox(string sRect, string sText) +{ + string sDraw = + nuiString("rect") + ":" + sRect + "," + + nuiString("text") + ":" + (nuiIsBind(sText) ? sText : nuiString(sText)) + "," + + nuiString("type") + ":" + nuiInt(4); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawImage(string sResref, string sRect, int nAspect, int nHAlign, int nVAlign) +{ + string sDraw = + nuiString("rect") + ":" + sRect + "," + + nuiString("image") + ":" + + (nuiIsBind(sResref) ? sResref : nuiString(sResref)) + "," + + nuiString("image_aspect") + ":" + nuiInt(nAspect) + "," + + nuiString("image_halign") + ":" + nuiInt(nHAlign) + "," + + nuiString("image_valign") + ":" + nuiInt(nVAlign) + "," + + nuiString("type") + ":" + nuiInt(5); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawArc(string sCenter, float fRadius, float fAMin, float fAMax) +{ + string sDraw = + nuiString("c") + ":" + sCenter + "," + + nuiString("radius") + ":" + nuiFloat(fRadius) + "," + + nuiString("amin") + ":" + nuiFloat(fAMin) + "," + + nuiString("amax") + ":" + nuiFloat(fAMax) + "," + + nuiString("type") + ":" + nuiInt(3); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_DrawCurve(string sA, string sB, string sCtrl0, string sCtrl1) +{ + string sDraw = + nuiString("a") + ":" + sA + "," + + nuiString("b") + ":" + sB + "," + + nuiString("ctrl0") + ":" + sCtrl0 + "," + + nuiString("ctrl1") + ":" + sCtrl1 + "," + + nuiString("type") + ":" + nuiInt(1); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_BindLine(string sBind) {NUI_DrawLine(nuiBind(sBind));} +void NUI_BindCircle(string sBind) {NUI_DrawDefinedCircle(nuiBind(sBind));} + +void NUI_BindTextbox(string sRectangle, string sText) +{ + string sDraw = + nuiString("text") + ":" + nuiBind(sText) + "," + + nuiString("points") + ":" + nuiBind(sRectangle) + "," + + nuiString("type") + ":" + nuiInt(4); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_BindImage(string sResref, string sRectangle, string sAspect, string sHAlign, string sVAlign) +{ + string sDraw = + nuiString("rect") + ":" + nuiBind(sRectangle) + "," + + nuiString("image") + ":" + nuiBind(sResref) + "," + + nuiString("image_aspect") + ":" + nuiBind(sAspect) + "," + + nuiString("image_halign") + ":" + nuiBind(sHAlign) + "," + + nuiString("image_valign") + ":" + nuiBind(sVAlign) + "," + + nuiString("type") + ":" + nuiInt(5); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_BindArc(string sCenter, string sRadius, string sStartAngle, string sEndAngle) +{ + string sDraw = + nuiString("c") + ":" + nuiBind(sCenter) + "," + + nuiString("radius") + ":" + nuiBind(sRadius) + "," + + nuiString("amin") + ":" + nuiBind(sStartAngle) + "," + + nuiString("amax") + ":" + nuiBind(sEndAngle) + "," + + nuiString("type") + ":" + nuiInt(3); + + nui_ToggleIncrementFlag(FALSE); + nui_SetControl(nui_CreateCanvasTemplate(sDraw)); +} + +void NUI_BindCurve(string sStart, string sEnd, string sCtrl0, string sCtrl1) +{ + NUI_DrawCurve(nuiBind(sStart), nuiBind(sEnd), nuiBind(sCtrl0), nuiBind(sCtrl1)); +} + +// ----------------------------------------------------------------------------- +// Form and Control Properties +// ----------------------------------------------------------------------------- + +// Binds ----------------------------------------------------------------------- +// Forms --------------------------------------------------------------------- +void NUI_BindAcceptsInput(string sBind, int bWatch = FALSE) {nui_SetProperty("accepts_input", nuiBind(sBind, bWatch));} +void NUI_BindClosable(string sBind, int bWatch = FALSE) {nui_SetProperty("closable", nuiBind(sBind, bWatch));} +void NUI_BindCollapsible(string sBind, int bWatch = FALSE) {nui_SetProperty("collapsed", nuiBind(sBind, bWatch));} +void NUI_BindGeometry(string sBind, int bWatch = FALSE) {nui_SetProperty("geometry", nuiBind(sBind, bWatch));} +void NUI_BindResizable(string sBind, int bWatch = FALSE) {nui_SetProperty("resizable", nuiBind(sBind, bWatch));} +void NUI_BindTitle(string sBind, int bWatch = FALSE) {nui_SetProperty("title", nuiBind(sBind, bWatch));} +void NUI_BindTitleColor(string sBind, int bWatch = FALSE) {nui_SetProperty("foreground_color", nuiBind(sBind, bWatch));} +void NUI_BindTransparent(string sBind, int bWatch = FALSE) {nui_SetProperty("transparent", nuiBind(sBind, bWatch));} + +// Binds ----------------------------------------------------------------------- +// Shared -------------------------------------------------------------------- +void NUI_BindBorder(string sBind, int bWatch = FALSE) {nui_SetProperty("border", nuiBind(sBind, bWatch));} + +// Binds ----------------------------------------------------------------------- +// Controls ------------------------------------------------------------------ +void NUI_BindData(string sBind, int bWatch = FALSE) {nui_SetProperty("data", nuiBind(sBind, bWatch));} +void NUI_BindDisabledTooltip(string sBind, int bWatch = FALSE) {nui_SetProperty("disabled_tooltip", nuiBind(sBind, bWatch));} +void NUI_BindElements(string sBind, int bWatch = FALSE) {nui_SetProperty("elements", nuiBind(sBind, bWatch));} +void NUI_BindEnabled(string sBind, int bWatch = FALSE) {nui_SetProperty("enabled", nuiBind(sBind, bWatch));} +void NUI_BindEncouraged(string sBind, int bWatch = FALSE) {nui_SetProperty("encouraged", nuiBind(sBind, bWatch));} +void NUI_BindEndPoint(string sBind, int bWatch = FALSE) {nui_SetProperty("b", nuiBind(sBind, bWatch));} +void NUI_BindFill(string sBind, int bWatch = FALSE) {nui_SetProperty("fill", nuiBind(sBind, bWatch));} +void NUI_BindForegroundColor(string sBind, int bWatch = FALSE) {nui_SetProperty("foreground_color", nuiBind(sBind, bWatch));} +void NUI_BindHorizontalAlignment(string sBind, int bWatch = FALSE) {nui_SetProperty("text_halign", nuiBind(sBind, bWatch));} +void NUI_BindLegend(string sBind, int bWatch = FALSE) {nui_SetProperty("legend", nuiBind(sBind, bWatch));} +void NUI_BindLength(string sBind, int bWatch = FALSE) {nui_SetProperty("max", nuiBind(sBind, bWatch));} +void NUI_BindLineThickness(string sBind, int bWatch = FALSE) {nui_SetProperty("line_thickness", nuiBind(sBind, bWatch));} +void NUI_BindPlaceholder(string sBind, int bWatch = FALSE) {nui_SetProperty("label", nuiBind(sBind, bWatch));} +void NUI_BindPoints(string sBind, int bWatch = FALSE) {nui_SetProperty("points", nuiBind(sBind, bWatch));} +void NUI_BindRadius(string sBind, int bWatch = FALSE) {nui_SetProperty("radius", nuiBind(sBind, bWatch));} +void NUI_BindRectangle(string sBind, int bWatch = FALSE) {nui_SetProperty("rect", nuiBind(sBind, bWatch));} +void NUI_BindRegion(string sBind, int bWatch = FALSE) {nui_SetProperty("image_region", nuiBind(sBind, bWatch));} +void NUI_BindRowCount(string sBind, int bWatch = FALSE) {nui_SetProperty("row_count", nuiBind(sBind, bWatch));} +void NUI_BindScissor(string sBind, int bWatch = FALSE) {nui_SetProperty("draw_list_scissor", nuiBind(sBind, bWatch));} +void NUI_BindStartPoint(string sBind, int bWatch = FALSE) {nui_SetProperty("a", nuiBind(sBind, bWatch));} +void NUI_BindStep(string sBind, int bWatch = FALSE) {nui_SetProperty("step", nuiBind(sBind, bWatch));} +void NUI_BindText(string sBind, int bWatch = FALSE) {nui_SetProperty("text", nuiBind(sBind, bWatch));} +void NUI_BindType(string sBind, int bWatch = FALSE) {nui_SetProperty("type", nuiBind(sBind, bWatch));} +void NUI_BindValue(string sBind, int bWatch = FALSE) {nui_SetProperty("value", nuiBind(sBind, bWatch));} +void NUI_BindVerticalAlignment(string sBind, int bWatch = FALSE) {nui_SetProperty("text_valign", nuiBind(sBind, bWatch));} +void NUI_BindVisible(string sBind, int bWatch = FALSE) {nui_SetProperty("visible", nuiBind(sBind, bWatch));} + +void NUI_BindAspect(string sBind, int bWatch = FALSE) +{ + string sProperty; + if (nui_GetDrawlistFlag() || nui_GetControlType() == "image") + sProperty = "image_aspect"; + else + sProperty = "aspect"; + + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindColor(string sBind, int bWatch = FALSE) +{ + string sProperty = (nui_GetControlType() == "color_picker" ? "value" : "color"); + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindSliderBounds(string sUpper, string sLower, string sStep, int bWatch = FALSE) +{ + nui_SetProperty("max", nuiBind(sUpper, bWatch)); + nui_SetProperty("min", nuiBind(sLower, bWatch)); + nui_SetProperty("step", nuiBind(sStep, bWatch)); +} + +void NUI_BindLabel(string sBind, int bWatch = FALSE) +{ + string sProperty = (nui_GetControlType() == "label" ? "value" : "label"); + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindMax(string sBind, int bWatch = FALSE) +{ + string sProperty = (nui_GetDrawlistFlag() ? "amax" : "max"); + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindMin(string sBind, int bWatch = FALSE) +{ + string sProperty = (nui_GetDrawlistFlag() ? "amin" : "min"); + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindResref(string sBind, int bWatch = FALSE) +{ + string sProperty, sType = nui_GetControlType(); + if (sType == "button_image") sProperty = "label"; + else if (sType == "image") sProperty = "value"; + else if (sType == "movieplayer") sProperty = "value"; + else sProperty = "image"; + + nui_SetProperty(sProperty, nuiBind(sBind, bWatch)); +} + +void NUI_BindTooltip(string sBind, int bDisabledTooltip = FALSE, int bWatch = FALSE) +{ + if (nui_GetControlType() == "combo") + return; + + nui_SetProperty("tooltip", nuiBind(sBind, bWatch)); + if (bDisabledTooltip) NUI_BindDisabledTooltip(sBind, bWatch); +} + +// Sets ------------------------------------------------------------------------ +// Forms --------------------------------------------------------------------- +void NUI_SetAcceptsInput(int bAcceptsInput = TRUE) {nui_SetProperty("accepts_input", nuiBool(bAcceptsInput));} +void NUI_SetClosable(int bClosable = TRUE) {nui_SetProperty("closable", nuiBool(bClosable));} +void NUI_SetCollapsible(int bCollapsible = TRUE) {nui_SetProperty("collapsed", nuiBool(bCollapsible));} +void NUI_SetTitle(string sTitle) {nui_SetProperty("title", nuiString(sTitle));} +void NUI_SetTitleColor(string sColor) {nui_SetProperty("foreground_color", sColor);} +void NUI_SetTOCTitle(string sTitle) {nui_SetProperty("toc_title", nuiString(sTitle));} +void NUI_SetDefinedGeometry(string sGeometry) {nui_SetProperty("geometry", sGeometry);} +void NUI_SetGeometry(float x, float y, float w, float h) {nui_SetProperty("geometry", NUI_DefineRectangle(x, y, w, h));} +void NUI_SetResizable(int bResizable = TRUE) {nui_SetProperty("resizable", nuiBool(bResizable));} +void NUI_SetTransparent(int bTransparent = TRUE) {nui_SetProperty("transparent", nuiBool(bTransparent));} + +// Sets ------------------------------------------------------------------------ +// Shared -------------------------------------------------------------------- +void NUI_SetBorder(int bVisible = TRUE) {nui_SetProperty("border", nuiBool(bVisible));} + +// Sets ------------------------------------------------------------------------ +// Controls ------------------------------------------------------------------ +void NUI_SetAspect(int nAspect) {nui_SetProperty("image_aspect", nuiInt(nAspect));} +void NUI_SetAspectRatio(float fAspect) {nui_SetProperty("aspect", nuiFloat(fAspect));} +void NUI_SetCenter(float x, float y) {nui_SetProperty("c", NUI_DefinePoint(x, y));} +void NUI_SetDefaultValue(string sDefault) {nui_SetProperty("default_value", sDefault);} +void NUI_SetDisabledTooltip(string sTooltip) {nui_SetProperty("disabled_tooltip", nuiString(sTooltip));} +void NUI_SetDirection(int nDirection = NUI_ORIENTATION_ROW) {nui_SetProperty("direction", nuiInt(nDirection));} +void NUI_SetDrawCondition(int nCondition = NUI_DRAW_ALWAYS) {nui_SetProperty("render", nuiInt(nCondition));} +void NUI_SetDrawPosition(int nPosition = NUI_DRAW_ABOVE) {nui_SetProperty("order", nuiInt(nPosition));} +void NUI_SetEnabled(int bEnabled = TRUE) {nui_SetProperty("enabled", nuiBool(bEnabled));} +void NUI_SetEncouraged(int bEncouraged = TRUE) {nui_SetProperty("encouraged", nuiBool(bEncouraged));} +void NUI_SetFill(int bFill = TRUE) {nui_SetProperty("fill", nuiBool(bFill));} +void NUI_SetForegroundColor(string sColor) {nui_SetProperty("foreground_color", sColor);} +void NUI_SetHeight(float fHeight) {nui_SetProperty("height", nuiFloat(fHeight));} +void NUI_SetID(string sID) {nui_SetProperty("id", nuiString(sID));} +void NUI_SetLength(int nLength) {nui_SetProperty("max", nuiInt(nLength));} +void NUI_SetLineThickness(float fThickness) {nui_SetProperty("line_thickness", nuiFloat(fThickness));} +void NUI_SetMargin(float fMargin) {nui_SetProperty("margin", nuiFloat(fMargin));} +void NUI_SetMultiline(int bMultiline = TRUE) {nui_SetProperty("multiline", nuiBool(bMultiline));} +void NUI_SetPadding(float fPadding) {nui_SetProperty("padding", nuiFloat(fPadding));} +void NUI_SetPlaceholder(string sText) {nui_SetProperty("label", nuiString(sText));} +void NUI_SetPoints(string sPoints) {nui_SetProperty("points", sPoints);} +void NUI_SetRadius(float r) {nui_SetProperty("radius", nuiFloat(r));} +void NUI_SetRectangle(string sRectangle) {nui_SetProperty("rect", sRectangle);} +void NUI_SetRegion(string sRegion) {nui_SetProperty("image_region", sRegion);} +void NUI_SetRowCount(int nRowCount) {nui_SetProperty("row_count", nuiInt(nRowCount));} +void NUI_SetRowHeight(float fRowHeight) {nui_SetProperty("row_height", nuiFloat(fRowHeight));} +void NUI_SetScissor(int bScissor) {nui_SetProperty("draw_list_scissor", nuiBool(bScissor));} +void NUI_SetScrollbars(int nScrollbars = NUI_SCROLLBARS_AUTO) {nui_SetProperty("scrollbars", nuiInt(nScrollbars));} +void NUI_SetStatic() {nui_SetProperty("type", nuiString("text"));} +void NUI_SetTemplateVariable(int bVariable) {nui_SetProperty("NUI_TEMPLATE_VARIABLE", nuiBool(bVariable));} +void NUI_SetTemplateWidth(float fWidth) {nui_SetProperty("NUI_TEMPLATE_WIDTH", nuiFloat(fWidth));} +void NUI_SetText(string sText) {nui_SetProperty("text", nuiString(sText));} +void NUI_SetValue(string sValue) {nui_SetProperty("value", sValue);} +void NUI_SetVisible(int bVisible = TRUE) {nui_SetProperty("visible", nuiBool(bVisible));} +void NUI_SetWidth(float fWidth) {nui_SetProperty("width", nuiFloat(fWidth));} +void NUI_SetWordWrap(int bWrap = TRUE) {nui_SetProperty("wordwrap", nuiBool(bWrap));} + +void NUI_SetColor(string sColor) +{ + string sProperty = (nui_GetControlType() == "color_picker" ? "value" : "color"); + nui_SetProperty(sProperty, sColor); +} + +void NUI_SetDimensions(float fWidth, float fHeight) +{ + NUI_SetWidth(fWidth); + NUI_SetHeight(fHeight); +} + +void NUI_SetElements(string sElements) +{ + int nEntry = nui_GetEntryCount(); + string sType = nui_GetControlType(); + int n; for(n; n < CountList(sElements); n++) + { + string sElement, sEntry = GetListItem(sElements, n); + if (sType == "combo") + sElement = "[" + + nuiString(sEntry) + "," + + nuiInt(nEntry++) + + "]"; + else if (sType == "options" || sType == "tabbar") + sElement = nuiString(sEntry); + + nui_IncrementEntryCount(); + nui_SetProperty("NUI_ELEMENT", sElement); + } +} + +void NUI_SetFloatSliderBounds(float fLower, float fUpper, float fStep) +{ + nui_SetProperty("min", nuiFloat(fLower)); + nui_SetProperty("max", nuiFloat(fUpper)); + nui_SetProperty("step", nuiFloat(fStep)); +} + +void NUI_SetHorizontalAlignment(int nAlign) +{ + string sProperty = (nui_GetControlType() == "label" ? "text_halign" : "image_halign"); + nui_SetProperty(sProperty, nuiInt(nAlign)); +} + +void NUI_SetIntSliderBounds(int nLower, int nUpper, int nStep) +{ + nui_SetProperty("min", nuiInt(nLower)); + nui_SetProperty("max", nuiInt(nUpper)); + nui_SetProperty("step", nuiInt(nStep)); +} + +void NUI_SetLabel(string sLabel) +{ + string sProperty = (nui_GetControlType() == "label" ? "value" : "label"); + nui_SetProperty(sProperty, nuiString(sLabel)); +} + +void NUI_SetResref(string sResref) +{ + string sProperty, sType = nui_GetControlType(); + if (sType == "button_image") sProperty = "label"; + else if (sType == "image") sProperty = "value"; + else if (sType == "movieplayer") sProperty = "value"; + else if (sType == "button") sProperty = "value"; + else sProperty = "image"; + + nui_SetProperty(sProperty, nuiString(sResref)); +} + +void NUI_SetSquare(float fSide) +{ + NUI_SetHeight(fSide); + NUI_SetWidth(fSide); +} + +void NUI_SetTooltip(string sTooltip, int bDisabledTooltip = FALSE) +{ + if (nui_GetControlType() == "combo") + return; + + nui_SetProperty("tooltip", nuiString(sTooltip)); + if (bDisabledTooltip) NUI_SetDisabledTooltip(sTooltip); +} + +// TODO alignments for future movie player growth? +void NUI_SetVerticalAlignment(int nAlign) +{ + string sProperty = (nui_GetControlType() == "label" ? "text_valign" : "image_valign"); + nui_SetProperty(sProperty, nuiInt(nAlign)); +} + +// TODO not tested +void NUI_SetChartSeries(int nType, string sLegend, string sColor, string sData) +{ + string sChart = "{" + + nuiString("type") + ":" + nuiInt(nType) + "," + + nuiString("legend") + ":" + nuiString(sLegend) + "," + + nuiString("color") + ":" + sColor + "," + + nuiString("data") + ":" + sData + + "}"; + + nui_SetProperty("NUI_SERIES", sChart); +} + +// ----------------------------------------------------------------------------- +// Form Definition/Management +// Private +// ----------------------------------------------------------------------------- + +json nui_GetForm(string sFormID, int bForceModule = FALSE) +{ + sql = nui_PrepareQuery("SELECT definition FROM nui_forms WHERE form = @form;", bForceModule); + SqlBindString(sql, "@form", sFormID); + + return SqlStep(sql) ? SqlGetJson(sql, 0) : JsonNull(); +} + +string nui_GetFormsByPrefix(string sForms, string sPrefix, int nResType) +{ + string sForm; + int n; while ((sForm = ResManFindPrefix(sPrefix, nResType, ++n)) != "") + sForms = AddListItem(sForms, sForm, TRUE); + + return sForms; +} + +string nui_GetForms(string sPrefix) +{ + string sForms = nui_GetFormsByPrefix("", sPrefix, RESTYPE_NCS); + return sForms = nui_GetFormsByPrefix(sForms, sPrefix, RESTYPE_NSS); +} + +int nui_ExecuteFunction(string sFile, string sFunction, object oTarget = OBJECT_SELF, string sArguments = "") +{ + if (sFile == "" || sFunction == "") + return FALSE; + + if (ResManFindPrefix(sFile, RESTYPE_NCS) == sFile) + { + SetScriptParam("NUI_FUNCTION", sFunction); + SetScriptParam("NUI_ARGS", sArguments); + ExecuteScript(sFile, oTarget); + return TRUE; + } + + if (sArguments != "") sArguments = nuiString(sArguments); + + string sChunk = "#" + "include " + nuiString(sFile) + " " + + "void main() {" + sFunction + "(" + sArguments + ");}"; + return ExecuteScriptChunk(sChunk, oTarget, FALSE) == ""; +} + +json nui_GetWatchedBinds(string sFormID) +{ + sQuery = "SELECT json_group_array(value) FROM (SELECT DISTINCT value FROM nui_forms, " + + "json_tree(nui_forms.definition, '$') WHERE key = 'bind' AND json_extract(" + + "nui_forms.definition, path || '.watch') = true AND form = @form);"; + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sFormID); + return SqlStep(sql) ? SqlGetJson(sql, 0) : JsonArray(); +} + +json NUI_GetOrphanBinds(string sFormID) +{ + sQuery = "SELECT json_group_array(value) FROM (SELECT DISTINCT value FROM nui_forms, " + + "json_tree(nui_forms.definition, '$') WHERE key = 'bind' and form = @form EXCEPT " + + "SELECT key FROM (SELECT DISTINCT key FROM nui_forms, json_each(nui_forms.definition, " + + "'$.profiles.default') AS value WHERE form = @form));"; + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sFormID); + return SqlStep(sql) ? SqlGetJson(sql, 0) : JsonArray(); +} + +// ----------------------------------------------------------------------------- +// Form Profiles +// Private +// ----------------------------------------------------------------------------- + +void nui_SetProfile(string sProfile) {SetLocalString(nui_GetDataObject(), "NUI_PROFILE", sProfile);} +string nui_GetProfile() {return GetLocalString(nui_GetDataObject(), "NUI_PROFILE");} + +/// @private Sets the profile bind value into the form definition. +void nui_SetProfileBind(string sProperty, string sJson = "") +{ + string sPath = "$.profiles." + nui_GetProfile() + "." + sProperty; + sQuery = "UPDATE nui_forms SET definition = (SELECT json_set(definition, '" + sPath + + "', json(@json)) FROM nui_forms WHERE form = @form) WHERE form = @form;"; + + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", nui_GetFormID()); + SqlBindString(sql, "@json", sJson); + SqlStep(sql); +} + +/// @private Used when basing a new profile off an old profile. To prevent massive amounts +/// of potential recursion, just copy the base profile and go from there. +void nui_CopyProfile(string sBase) +{ + sQuery = "WITH base AS (SELECT value FROM nui_forms, json_tree(nui_forms.definition, " + + "'$.profiles') WHERE key = @base AND form = @form) UPDATE nui_forms SET definition = " + + "(SELECT json_set(definition, '$.profiles." + nui_GetProfile() + "', json_extract(base.value, '$')) " + + "FROM nui_forms, base WHERE form = @form) WHERE form = @form;"; + + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@base", sBase); + SqlBindString(sql, "@form", nui_GetFormID()); + SqlStep(sql); +} + +/// @private Returns a json object of bind:value pairs from the default profile as modified +/// by sProfile. +json nui_GetProfileBinds(string sFormID, string sProfile = "") +{ + sQuery = "WITH def AS (SELECT value FROM nui_forms, json_tree(nui_forms.definition, " + + "'$.profiles') WHERE key = 'default' AND form = @form), sel AS (SELECT " + + "COALESCE((SELECT value from nui_forms, json_tree(nui_forms.definition, '$.profiles') " + + "WHERE key = @profile AND form = @form), json_object()) value FROM nui_forms WHERE form = @form) " + + "SELECT json_patch(json_extract(def.value, '$'), json_extract(sel.value, '$')) FROM def, sel;"; + + sql = nui_PrepareQuery(sQuery); + SqlBindString(sql, "@form", sFormID); + SqlBindString(sql, "@profile", sProfile); + return SqlStep(sql) ? SqlGetJson(sql, 0) : JsonObject(); +} + +/// @private Called during form opening, sets the initial values for all default binds. +void nui_SetProfileBinds(object oPC, string sFormID, string sProfile) +{ + json jProfile = nui_GetProfileBinds(sFormID, sProfile); + json jKeys = JsonObjectKeys(jProfile); + + int n; for (n; n < JsonGetLength(jKeys); n++) + { + string sKey = JsonGetString(JsonArrayGet(jKeys, n)); + NUI_SetBindJ(oPC, sFormID, sKey, JsonObjectGet(jProfile, sKey)); + } + + NUI_SetBind(oPC, sFormID, "NUI_FORM_PROFILE", nuiString(sProfile)); +} + +void nui_SetWatchedBinds(object oPC, string sFormID) +{ + json jBinds = nui_GetWatchedBinds(sFormID); + int n; for (n; n < JsonGetLength(jBinds); n++) + { + string sBind = JsonGetString(JsonArrayGet(jBinds, n)); + NUI_SetBindWatch(oPC, sFormID, sBind); + } +} + +// ----------------------------------------------------------------------------- +// Form Definition/Management +// Public +// ----------------------------------------------------------------------------- + +void NUI_DefineForms(string sFormfile = "") +{ + nui_ClearVariables(); + nui_BeginTransaction(); + nui_ToggleDefinitionFlag(TRUE); + + if (sFormfile != "") + { + nui_SetFormfile(sFormfile); + nui_ExecuteFunction(sFormfile, NUI_DEFINE); + } + else + { + int n; for (n; n < CountList(NUI_FORMFILE_PREFIX); n++) + { + string sFormfiles = nui_GetForms(GetListItem(NUI_FORMFILE_PREFIX, n)); + if (sFormfiles == "") return; + + int f; for (f; f < CountList(sFormfiles); f++) + { + sFormfile = GetListItem(sFormfiles, f); + + nui_SetFormfile(sFormfile); + nui_ExecuteFunction(sFormfile, NUI_DEFINE); + } + } + } + + nui_ToggleDefinitionFlag(FALSE); + nui_CopyDefinitions(); + nui_CommitTransaction(); +} + +int NUI_DisplayForm(object oPC, string sFormID, string sProfile = "default") +{ + json jForm = nui_GetForm(sFormID); + if (jForm != JsonNull()) + { + int nToken = NuiCreate(oPC, jForm, sFormID); + json jData = JsonObjectSet(JsonObject(), "profile", JsonString(sProfile)); + jData = JsonObjectSet(jData, "formfile", JsonString(nui_GetDefinitionValue(sFormID, "formfile"))); + + NuiSetUserData(oPC, NuiFindWindow(oPC, sFormID), jData); + return nToken; + } + + return -1; +} + +void NUI_CloseForm(object oPC, string sFormID) {NuiDestroy(oPC, NuiFindWindow(oPC, sFormID));} + +void NUI_DisplaySubform(object oPC, string sFormID, string sElementID, string sSubformID) +{ + int nToken = NuiFindWindow(oPC, sFormID); + string sLayout = nui_GetDefinitionValue(NuiGetWindowId(oPC, nToken), "subforms." + sSubformID); + NuiSetGroupLayout(oPC, nToken, sElementID, JsonParse(sLayout)); +} + +int NUI_GetFormToken(object oPC, string sFormID) +{ + return NuiFindWindow(oPC, sFormID); +} + +// ----------------------------------------------------------------------------- +// Form Profiles +// Public +// ----------------------------------------------------------------------------- + +void NUI_CreateDefaultProfile() {NUI_CreateProfile("default");} + +void NUI_CreateProfile(string sProfile, string sBase = "") +{ + nui_SetProfile(sProfile); + + if (sBase != "") + nui_CopyProfile(sBase); +} + +void NUI_SetProfileBind(string sBind, string sJson) +{ + if (sBind == "" || sJson == "" || JsonParse(sJson) == JsonNull()) + return; + + nui_SetProfileBind(sBind, sJson); +} + +void NUI_SetProfileBindJ(string sBind, json jValue) +{ + NUI_SetProfileBind(sBind, JsonDump(jValue)); +} + +void NUI_SetProfileBinds(string sBinds, string sJson) +{ + if (sBinds == "" || sJson == "" || JsonParse(sJson) == JsonNull()) + return; + + int n; for (n; n < CountList(sBinds); n++) + nui_SetProfileBind(GetListItem(sBinds, n), sJson); +} + +void NUI_SetProfile(object oPC, string sFormID, string sProfile) +{ + nui_SetProfileBinds(oPC, sFormID, sProfile); +} + +string NUI_GetProfile(object oPC, string sFormID) +{ + return JsonGetString(NUI_GetBind(oPC, sFormID, "NUI_FORM_PROFILE")); +} + +// ----------------------------------------------------------------------------- +// Event Management +// Private +// ----------------------------------------------------------------------------- + +int nui_HandleInspectionEvents(int nEventID = -1) +{ + struct NUIEventData ed = NUI_GetEventData(); + + if (NUI_FI_RECORD_DATA == NUI_FI_NEVER) + return -1; + else if (NUI_FI_RECORD_DATA == NUI_FI_WHEN_OPEN && !NUI_GetIsFormOpen(ed.oPC, "nui_inspector")) + return -1; + + if (ed.sEvent == "close") + { + sql = nui_PrepareQuery("DELETE FROM nui_fi_data WHERE nToken = @token;"); + SqlBindInt(sql, "@token", NuiGetEventWindow()); + SqlStep(sql); + return -1; + } + + json jCurrentState = NUI_GetBindState(ed.oPC, ed.sFormID); + + string sQuery; + sqlquery sql; + + if (nEventID == -1) + { + sQuery = + "INSERT INTO nui_fi_data (sPlayer, sFormID, nToken, sEvent, sControlID, nIndex, jPayload, " + + "binds_before, timestamp) VALUES (@sPlayer, @sFormID, @nToken, @sEvent, @sControlID, @nIndex, " + + "@jPayload, @binds_before, strftime('%s','now')) RETURNING event_id;"; + + sql = nui_PrepareQuery(sQuery); + + SqlBindString(sql, "@sPlayer", GetObjectUUID(ed.oPC)); + SqlBindString(sql, "@sFormID", ed.sFormID); + SqlBindInt (sql, "@nToken", ed.nToken); + SqlBindString(sql, "@sEvent", ed.sEvent); + SqlBindString(sql, "@sControlID", ed.sControlID); + SqlBindInt (sql, "@nIndex", ed.nIndex); + SqlBindJson (sql, "@jPayload", ed.jPayload); + SqlBindJson (sql, "@binds_before", jCurrentState); + + nEventID = SqlStep(sql) ? SqlGetInt(sql, 0) : -1; + } + else + { + sQuery = + "UPDATE nui_fi_data SET binds_after = @binds_after " + + "WHERE event_id = @event_id;"; + sql = nui_PrepareQuery(sQuery); + SqlBindJson (sql, "@binds_after", jCurrentState); + SqlBindInt (sql, "@event_id", nEventID); + + SqlStep(sql); + } + + if (nEventID != -1) + { + SignalEvent(GetModule(), EventUserDefined(NUI_FI_EVENT_UPDATE_FORMS)); + SignalEvent(GetModule(), EventUserDefined(NUI_FI_EVENT_UPDATE_EVENTS)); + } + + return nEventID; +} + +void nui_HandleNUIEvents() +{ + object oPC = NuiGetEventPlayer(); + string sFormID = NuiGetWindowId(oPC, NuiGetEventWindow()); + + json jUserData = NuiGetUserData(oPC, NuiGetEventWindow()); + string sFormfile = JsonGetString(JsonObjectGet(jUserData, "formfile")); + + // TODO integrate form inspection + // Should we check for the inspection window, or just save everything? + // Or hide it behind a configuration option? + int nInspectorEventID = nui_HandleInspectionEvents(); + + string sEvent = NuiGetEventType(); + if (sEvent == "open") + { + string sProfile = JsonGetString(JsonObjectGet(jUserData, "profile")); + nui_SetProfileBinds(oPC, sFormID, sProfile); + nui_SetWatchedBinds(oPC, sFormID); + nui_ExecuteFunction(sFormfile, NUI_BIND, oPC); + } + + nui_ExecuteFunction(sFormfile, NUI_EVENT_NUI, oPC); + + if (nInspectorEventID != -1) + nui_HandleInspectionEvents(nInspectorEventID); + + +} + +// ----------------------------------------------------------------------------- +// Event Management +// Public +// ----------------------------------------------------------------------------- + +void NUI_SetBind(object oPC, string sFormID, string sBind, string sValue) +{ + NuiSetBind(oPC, NuiFindWindow(oPC, sFormID), sBind, JsonParse(sValue)); +} + +void NUI_SetBindJ(object oPC, string sFormID, string sBind, json jValue) +{ + NuiSetBind(oPC, NuiFindWindow(oPC, sFormID), sBind, jValue); +} + +void NUI_SetBindWatch(object oPC, string sFormID, string sBind, int bWatch = TRUE) +{ + NuiSetBindWatch(oPC, NuiFindWindow(oPC, sFormID), sBind, bWatch); +} + +json NUI_GetBind(object oPC, string sFormID, string sBind) +{ + return NuiGetBind(oPC, NuiFindWindow(oPC, sFormID), sBind); +} + +int NUI_GetIsFormOpen(object oPC, string sFormID) +{ + return NuiFindWindow(oPC, sFormID) > 0; +} + +struct NUIEventData NUI_GetEventData() +{ + struct NUIEventData ed; + + ed.oPC = NuiGetEventPlayer(); + ed.nToken = NuiGetEventWindow(); + ed.sFormID = NuiGetWindowId(ed.oPC, ed.nToken); + ed.sEvent = NuiGetEventType(); + ed.sControlID = NuiGetEventElement(); + ed.nIndex = NuiGetEventArrayIndex(); + ed.jPayload = NuiGetEventPayload(); + + return ed; +} + +void NUI_SubscribeEvent(int nEvent) +{ + sQuery = "UPDATE nui_forms SET definition = (SELECT json_set(definition, " + + "'$.event_data[#]', json(@value)) FROM nui_forms WHERE form = @form) " + + "WHERE form = @form;"; + + sql = nui_PrepareQuery(sQuery, TRUE); + SqlBindString(sql, "@form", nui_GetFormID()); + SqlBindInt (sql, "@value", nEvent); + SqlStep(sql); +} + +void NUI_HandleEvents(object oPC = OBJECT_SELF) +{ + int nEvent = GetCurrentlyRunningEvent(); + + if (nEvent == EVENT_SCRIPT_MODULE_ON_NUI_EVENT) + nui_HandleNUIEvents(); + else + { + sQuery = "SELECT json_group_array(json_extract(nui_forms.definition, '$.formfile')) " + + "FROM nui_forms WHERE EXISTS (SELECT 1 FROM json_each(nui_forms.definition, " + + "'$.event_data') WHERE value = @event);"; + sql = nui_PrepareQuery(sQuery); + SqlBindInt(sql, "@event", nEvent); + + SetLocalObject(oPC, NUI_OBJECT, OBJECT_SELF); + + if (SqlStep(sql)) + { + json jFormfiles = SqlGetJson(sql, 0); + int n; for (n; n < JsonGetLength(jFormfiles); n++) + nui_ExecuteFunction(JsonGetString(JsonArrayGet(jFormfiles, n)), NUI_EVENT_MOD, oPC); + } + + DeleteLocalObject(oPC, NUI_OBJECT); + } +} diff --git a/_module/nss/pc_room_onexit.nss b/_module/nss/pc_room_onexit.nss new file mode 100644 index 0000000..f52e452 --- /dev/null +++ b/_module/nss/pc_room_onexit.nss @@ -0,0 +1,15 @@ +void main() +{ + object oPlayer = GetExitingObject(); + +/* int nToken = NuiFindWindow(oPlayer, PC_NUI_WINDOW_ID); + +//:: Murder the poor window + + NuiDestroy(oPlayer, nToken); + + nToken = NuiFindWindow(oPlayer, PC_NUI_GOLD_WINDOW_ID); + if (nToken) + NuiDestroy(oPlayer, nToken); */ + +} diff --git a/_module/nss/ra_close_door.nss b/_module/nss/ra_close_door.nss new file mode 100644 index 0000000..f5d82c1 --- /dev/null +++ b/_module/nss/ra_close_door.nss @@ -0,0 +1,6 @@ +//Closes door if it is open +void main() +{ + DelayCommand(13.0f,ActionCloseDoor(OBJECT_SELF)); +} + diff --git a/_module/nss/se_door_death.nss b/_module/nss/se_door_death.nss new file mode 100644 index 0000000..8752df7 --- /dev/null +++ b/_module/nss/se_door_death.nss @@ -0,0 +1,86 @@ +//:://////////////////////////////////////////////////////////////////////////// +//:: Name Respawn Door/s v1.1 +//:: FileName se_door_death +//:: Copyright (c) 2001 Bioware Corp. +//:://////////////////////////////////////////////////////////////////////////// +/* + Respawn a door after a set delay + Set a float on the door ie. RESPAWN_TIMER = xx else the default is used + Thanks to Craig Welburn for the insight + Cheers to Zarathustra217 +*/ +//:://////////////////////////////////////////////////////////////////////////// +//:: Created By: Sir Elric +//:: Created On: 8th May, 2006 +//:: Modified On: 16th August, 2007 +//:: Event Used: OnDeath event of a door +//:://////////////////////////////////////////////////////////////////////////// + +#include "x2_inc_compon" + +// ----------------------------------------------------------------------------- +// CONSTANTS - Settings below +// ----------------------------------------------------------------------------- + +const float RESPAWN_TIMER_DEFAULT = 600.0; // Default respawn delay +const int DO_CRAFT_DROP = TRUE; // Drop default Bioware crafting item? + + +// ----------------------------------------------------------------------------- +// PROTOTYPES +// ----------------------------------------------------------------------------- + +// Respawn a door after a set delay +// If no float is set on the door ie. RESPAWN_TIMER = xx +// The default delay will be used ie. RESPAWN_TIMER_DEFAULT = xx +// oSelf: - Object calling the script +void SE_RespawnDoor(object oSelf); + + +// ----------------------------------------------------------------------------- +// FUNCTIONS +// ----------------------------------------------------------------------------- + +void SE_RespawnDoor(object oSelf) +{ + PlayAnimation(ANIMATION_DOOR_CLOSE); + int nHealAmount = GetMaxHitPoints(oSelf) - GetCurrentHitPoints(oSelf); + ApplyEffectToObject(DURATION_TYPE_INSTANT, EffectHeal(nHealAmount), oSelf); +} + + +// ----------------------------------------------------------------------------- +// MAIN +// ----------------------------------------------------------------------------- + + +void main() + { + object oSelf = OBJECT_SELF; + object oKiller = GetLastKiller(); + + SetIsDestroyable(FALSE); + + float fDelay = GetLocalFloat(oSelf, "RESPAWN_TIMER"); + if(fDelay == 0.0) + fDelay = RESPAWN_TIMER_DEFAULT; + + DelayCommand(fDelay, SE_RespawnDoor(oSelf)); + + if (!GetIsPC(oKiller)) + return; + + while (GetIsObjectValid(GetMaster(oKiller))) + { + oKiller=GetMaster(oKiller); + } + + if(GetIsObjectValid(oKiller)) + { + AdjustReputation(oKiller, OBJECT_SELF, -35); + AdjustAlignment (oKiller, ALIGNMENT_CHAOTIC, 10); + } + + if(DO_CRAFT_DROP) + craft_drop_placeable(); + } diff --git a/_module/nss/sei_sit.nss b/_module/nss/sei_sit.nss new file mode 100644 index 0000000..f155316 --- /dev/null +++ b/_module/nss/sei_sit.nss @@ -0,0 +1,34 @@ +// +// NWSit +// +// Script to make the PC using the object sit on it. +// +// (c) Shir'le E. Illios, 2002 (shirle@drowwanderer.com) +// +//////////////////////////////////////////////////////// + +void main() +{ + + // Get the character using the object. + object oPlayer = GetLastUsedBy(); + + // Make certain that the character is a PC. + if( GetIsPC( oPlayer ) ) + { + + // Get the object being sat on. + object oChair = OBJECT_SELF; + + // If the object is valid and nobody else is currently sitting on it. + if( GetIsObjectValid( oChair ) && + !GetIsObjectValid( GetSittingCreature( oChair ) ) ) + { + // Let the player sit on the object. + AssignCommand( oPlayer, ActionSit( oChair ) ); + } + + } // End if + +} // End main + diff --git a/_module/nss/sob_examine002.nss b/_module/nss/sob_examine002.nss new file mode 100644 index 0000000..ecfa59a --- /dev/null +++ b/_module/nss/sob_examine002.nss @@ -0,0 +1,7 @@ +//Allow PC to see his description. +void main() +{ + object oPC=GetLastUsedBy(); + + AssignCommand(oPC,ActionExamine(OBJECT_SELF)); +} diff --git a/_module/nss/util_c_color.nss b/_module/nss/util_c_color.nss new file mode 100644 index 0000000..83851e3 --- /dev/null +++ b/_module/nss/util_c_color.nss @@ -0,0 +1,225 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_color.nss +/// @author Ed Burke (tinygiant98) +/// @brief Configuration file for util_i_color.nss. +/// @details +/// These color codes are used with the functions from util_i_color.nss. These +/// are hex color codes, the same as you'd use in web design and may other +/// areas, so they are easy to look up and to copy-paste into other programs. +/// +/// You can change the values of any of constants below, but do not change the +/// names of the constants themselves. You can also add your own constants for +/// use in your module. +/// +/// ## Acknowledgement +/// - Function colors copied from https://nwn.wiki/display/NWN1/Colour+Tokens. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// X11 Color Palette +// ----------------------------------------------------------------------------- + +// ----- Whites ---------------------------------------------------------------- +const int COLOR_AZURE = 0xf0ffff; +const int COLOR_BEIGE = 0xf5f5dc; +const int COLOR_BLUE_ALICE = 0xf0f8ff; +const int COLOR_HONEYDEW = 0xf0fff0; +const int COLOR_IVORY = 0xfffff0; +const int COLOR_LAVENDERBLUSH = 0xfff0f5; +const int COLOR_LINEN = 0xfaf0e6; +const int COLOR_MINTCREAM = 0xf5fffa; +const int COLOR_MISTYROSE = 0xffe4e1; +const int COLOR_OLDLACE = 0xfdf5e6; +const int COLOR_SEASHELL = 0xfff5ee; +const int COLOR_SNOW = 0xfffafa; +const int COLOR_WHITE = 0xffffff; +const int COLOR_WHITE_ANTIQUE = 0xfaebd7; +const int COLOR_WHITE_FLORAL = 0xfffaf0; +const int COLOR_WHITE_GHOST = 0xf8f8ff; +const int COLOR_WHITE_SMOKE = 0xf5f5f5; + +// ----- Blues ----------------------------------------------------------------- +const int COLOR_AQUA = 0x00ffff; +const int COLOR_AQUAMARINE = 0x7fffd4; +const int COLOR_BLUE = 0x0000ff; +const int COLOR_BLUE_CORNFLOWER = 0x6495ed; +const int COLOR_BLUE_DARK = 0x00008b; +const int COLOR_BLUE_DODGER = 0x1e90ff; +const int COLOR_BLUE_LIGHT = 0xadd8e6; +const int COLOR_BLUE_MEDIUM = 0x0000cd; +const int COLOR_BLUE_MIDNIGHT = 0x191970; +const int COLOR_BLUE_POWDER = 0xb0e0e6; +const int COLOR_BLUE_ROYAL = 0x4169e1; +const int COLOR_BLUE_SKY = 0x87ceeb; +const int COLOR_BLUE_SKY_DEEP = 0x00bfff; +const int COLOR_BLUE_SKY_LIGHT = 0x87cefa; +const int COLOR_BLUE_SLATE = 0x6a5acd; +const int COLOR_BLUE_SLATE_MEDIUM = 0x7b68ee; +const int COLOR_BLUE_STEEL = 0x4682b4; +const int COLOR_BLUE_STEEL_LIGHT = 0xb0c4de; +const int COLOR_CYAN = 0x00ffff; +const int COLOR_CYAN_LIGHT = 0xe0ffff; +const int COLOR_NAVY = 0x000080; +const int COLOR_TURQUOISE = 0x40e0d0; +const int COLOR_TURQUOISE_DARK = 0x00ced1; +const int COLOR_TURQUOISE_MEDIUM = 0x48d1cc; +const int COLOR_TURQUOISE_PALE = 0xafeeee; + +// ----- Browns ---------------------------------------------------------------- +const int COLOR_BISQUE = 0xffe4c4; +const int COLOR_BLANCHED_ALMOND = 0xffebcd; +const int COLOR_BROWN = 0xa52a2a; +const int COLOR_BROWN_LIGHT = 0xd0814b; +const int COLOR_BROWN_ROSY = 0xbc8f8f; +const int COLOR_BROWN_SADDLE = 0x8b4513; +const int COLOR_BROWN_SANDY = 0xf4a460; +const int COLOR_BURLYWOOD = 0xdeb887; +const int COLOR_CHOCOLATE = 0xd2691e; +const int COLOR_CORNSILK = 0xfff8dc; +const int COLOR_GOLDENROD = 0xdaa520; +const int COLOR_GOLDENROD_DARK = 0xb8860b; +const int COLOR_MAROON = 0x800000; +const int COLOR_PERU = 0xcd853f; +const int COLOR_SIENNA = 0xa0522d; +const int COLOR_TAN = 0xd2b48c; +const int COLOR_WHEAT = 0xf5deb3; +const int COLOR_WHITE_NAVAJO = 0xffdead; + +// ----- Purples --------------------------------------------------------------- +const int COLOR_BLUE_SLATE_DARK = 0x483d8b; +const int COLOR_BLUE_VIOLET = 0x8a2be2; +const int COLOR_FUCHSIA = 0xff00ff; +const int COLOR_INDIGO = 0x4b0082; +const int COLOR_LAVENDER = 0xe6e6fa; +const int COLOR_MAGENTA = 0xff00ff; +const int COLOR_MAGENTA_DARK = 0x8b008b; +const int COLOR_ORCHID = 0xda70d6; +const int COLOR_ORCHID_DARK = 0x9932cc; +const int COLOR_ORCHID_MEDIUM = 0xba55d3; +const int COLOR_PLUM = 0xdda0dd; +const int COLOR_PURPLE = 0x800080; +const int COLOR_PURPLE_MEDIUM = 0x9370d8; +const int COLOR_THISTLE = 0xd8bfd8; +const int COLOR_VIOLET = 0xee82ee; +const int COLOR_VIOLET_DARK = 0x9400d3; +const int COLOR_VIOLET_LIGHT = 0xf397f8; + +// ----- Oranges --------------------------------------------------------------- +const int COLOR_CORAL = 0xff7f50; +const int COLOR_ORANGE = 0xffa500; +const int COLOR_ORANGE_DARK = 0xff8c00; +const int COLOR_ORANGE_LIGHT = 0xf3b800; +const int COLOR_ORANGE_RED = 0xff4500; +const int COLOR_SALMON_LIGHT = 0xffa07a; +const int COLOR_TOMATO = 0xff6347; + +// ----- Reds ------------------------------------------------------------------ +const int COLOR_CORAL_LIGHT = 0xf08080; +const int COLOR_CRIMSON = 0xdc143c; +const int COLOR_FIREBRICK = 0xb22222; +const int COLOR_RED = 0xff0000; +const int COLOR_RED_DARK = 0x8b0000; +const int COLOR_RED_INDIAN = 0xcd5c4c; +const int COLOR_RED_LIGHT = 0xfa6155; +const int COLOR_SALMON = 0xfa8072; +const int COLOR_SALMON_DARK = 0xe9967a; + +// ----- Pinks ----------------------------------------------------------------- +const int COLOR_PINK = 0xffc0cb; +const int COLOR_PINK_DEEP = 0xff1493; +const int COLOR_PINK_HOT = 0xff69b4; +const int COLOR_PINK_LIGHT = 0xffb6c1; +const int COLOR_VIOLET_RED_MEDIUM = 0xc71585; +const int COLOR_VIOLET_RED_PALE = 0xdb7093; + +// ----- Grays ----------------------------------------------------------------- +const int COLOR_BLACK = 0x000000; +const int COLOR_GAINSBORO = 0xdcdcdc; +const int COLOR_GRAY = 0x808080; +const int COLOR_GRAY_DARK = 0xa9a9a9; +const int COLOR_GRAY_DIM = 0x696969; +const int COLOR_GRAY_LIGHT = 0xd3d3d3; +const int COLOR_GRAY_SLATE = 0x708090; +const int COLOR_GRAY_SLATE_DARK = 0x2f4f4f; +const int COLOR_GRAY_SLATE_LIGHT = 0x778899; +const int COLOR_SILVER = 0xc0c0c0; + +// ----- Greens ---------------------------------------------------------------- +const int COLOR_AQUAMARINE_MEDIUM = 0x66cdaa; +const int COLOR_CHARTREUSE = 0x7fff00; +const int COLOR_CYAN_DARK = 0x008b8b; +const int COLOR_GREEN = 0x008000; +const int COLOR_GREEN_DARK = 0x006400; +const int COLOR_GREEN_FOREST = 0x228b22; +const int COLOR_GREEN_LAWN = 0x7cfc00; +const int COLOR_GREEN_LIGHT = 0x90ee90; +const int COLOR_GREEN_LIME = 0x32cd32; +const int COLOR_GREEN_OLIVE_DARK = 0x556b2f; +const int COLOR_GREEN_PALE = 0x98fb98; +const int COLOR_GREEN_SEA = 0x2e8b57; +const int COLOR_GREEN_SEA_DARK = 0x8fbc8f; +const int COLOR_GREEN_SEA_LIGHT = 0x20b2aa; +const int COLOR_GREEN_SEA_MEDIUM = 0x3cb371; +const int COLOR_GREEN_SPRING = 0x00ff7f; +const int COLOR_GREEN_SPRING_MEDIUM = 0x00fa9a; +const int COLOR_GREEN_YELLOW = 0xadff2f; +const int COLOR_LIME = 0x00ff00; +const int COLOR_OLIVE = 0x808000; +const int COLOR_OLIVE_DRAB = 0x6b8e23; +const int COLOR_TEAL = 0x008080; +const int COLOR_YELLOW_GREEN = 0x9acd32; + +// ----- Yellows --------------------------------------------------------------- +const int COLOR_GOLD = 0xffd700; +const int COLOR_GOLDENROD_LIGHT = 0xfafad2; +const int COLOR_GOLDENROD_PALE = 0xeee8aa; +const int COLOR_KHAKI = 0xf0e68c; +const int COLOR_KHAKI_DARK = 0xbdb76b; +const int COLOR_LEMON_CHIFFON = 0xfffacd; +const int COLOR_MOCCASIN = 0xffe4b5; +const int COLOR_PAPAYA_WHIP = 0xffefd5; +const int COLOR_PEACH_PUFF = 0xffdab9; +const int COLOR_YELLOW = 0xffff00; +const int COLOR_YELLOW_DARK = 0xd0ce00; +const int COLOR_YELLOW_LIGHT = 0xffffe0; + +// ----------------------------------------------------------------------------- +// Colors By Function +// ----------------------------------------------------------------------------- + +const int COLOR_DEFAULT = 0xfefefe; +const int COLOR_ATTENTION = 0xfea400; +const int COLOR_BUG = 0x660000; +const int COLOR_FAIL = 0xff0000; +const int COLOR_SUCCESS = 0x3dc93d; +const int COLOR_DEBUG = 0xb4b4b4; +const int COLOR_INFO = 0xd0814b; + +// ----- Damage Types ---------------------------------------------------------- +const int COLOR_DAMAGE_MAGICAL = 0xcc77ff; +const int COLOR_DAMAGE_ACID = 0x01ff01; +const int COLOR_DAMAGE_COLD = 0x99ffff; +const int COLOR_DAMAGE_DIVINE = 0xffff01; +const int COLOR_DAMAGE_ELECTRICAL = 0x0166ff; +const int COLOR_DAMAGE_FIRE = 0xff0101; +const int COLOR_DAMAGE_NEGATIVE = 0x999999; +const int COLOR_DAMAGE_POSITIVE = 0xffffff; +const int COLOR_DAMAGE_SONIC = 0xff9901; + +// ----- Chat Log Messages ----------------------------------------------------- +const int COLOR_MESSAGE_FEEDBACK = 0xffff01; +const int COLOR_MESSAGE_COMBAT = 0xff6601; +const int COLOR_MESSAGE_MAGIC = 0xcc77ff; +const int COLOR_MESSAGE_SKILLS = 0x0166ff; +const int COLOR_MESSAGE_SAVING = 0x66ccff; +const int COLOR_SAVE_STATUS = 0x20ff20; +const int COLOR_PAUSE_STATE = 0xff0101; +const int COLOR_NAME_CLIENT = 0x99ffff; +const int COLOR_NAME_OTHER = 0xcc99cc; + +// ----------------------------------------------------------------------------- +// Custom Colors +// ----------------------------------------------------------------------------- +// You can add any custom colors you need for your functions below this line. +// ----------------------------------------------------------------------------- + diff --git a/_module/nss/util_c_debug.nss b/_module/nss/util_c_debug.nss new file mode 100644 index 0000000..e1ae9fb --- /dev/null +++ b/_module/nss/util_c_debug.nss @@ -0,0 +1,77 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_debug.nss +/// @author Ed Burke (tinygiant98) +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Configuration file for util_i_debug.nss. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Helper Constants and Functions +// ----------------------------------------------------------------------------- +// If you need custom constants, functions, or #includes, you may add them here. +// Since this file is included by util_i_debug.nss, you do not need to #include +// it to use its constants. +// ----------------------------------------------------------------------------- + +/* +// These constants are used by the example code below. Uncomment if you want to +// use them. + +/// @brief This is the minimum debug level required to trigger custom handling. +/// @details Setting this to DEBUG_LEVEL_ERROR means OnDebug() will handle only +/// messages marked as DEBUG_LEVEL_ERROR and DEBUG_LEVEL_CRITICAL. If set to +/// DEBUG_LEVEL_NONE, the user-defined event will never be triggered. +/// @warning It is not recommended to set this level to DEBUG_LEVEL_NOTICE or +/// DEBUG_LEVEL_DEBUG as this could create high message traffic rates. +const int DEBUG_EVENT_TRIGGER_LEVEL = DEBUG_LEVEL_ERROR; + +// These are varnames for script parameters +const string DEBUG_PARAM_PREFIX = "DEBUG_PARAM_PREFIX"; +const string DEBUG_PARAM_MESSAGE = "DEBUG_PARAM_MESSAGE"; +const string DEBUG_PARAM_LEVEL = "DEBUG_PARAM_LEVEL"; +const string DEBUG_PARAM_TARGET = "DEBUG_PARAM_TARGET"; +*/ + +// ----------------------------------------------------------------------------- +// Debug Handler +// ----------------------------------------------------------------------------- +// You may alter the contents of this function, but do not alter its signature. +// ----------------------------------------------------------------------------- + +/// @brief Custom debug event handler +/// @details This is a customizable function that runs before a message is shown +/// using Debug(). This function provides a user-definable hook into the +/// debug notification system. For example, a module can use this hook to +/// execute a script that may be able to handle module-specific error +/// handling or messaging. +/// @param sPrefix the debug message source provided by GetDebugPrefix() +/// @param sMessage the debug message provided to Debug() +/// @param nLevel the debug level of the message provided to Debug() +/// @param oTarget the game object being debugged as provided to Debug() +/// @returns TRUE if the message should be sent as normal; FALSE if no message +/// should be sent. +/// @note This function will never fire if oTarget is not debugging messages of +/// nLevel. +/// @warning Do not call Debug() or its aliases Notice(), Warning(), Error(), or +/// CriticalError() from this function; that would cause an infinite loop. +int HandleDebug(string sPrefix, string sMessage, int nLevel, object oTarget) +{ + /* + // The following example code allows an external script to handle the event + // with access to the appropriate script parameters. Optionally, all event + // handling can be accomplished directly in this function. + + // Only do custom handling if the debug level is error or critical error. + if (!nLevel || nLevel > DEBUG_EVENT_TRIGGER_LEVEL) + return TRUE; + + SetScriptParam(DEBUG_PARAM_PREFIX, sPrefix); + SetScriptParam(DEBUG_PARAM_MESSAGE, sMessage); + SetScriptParam(DEBUG_PARAM_LEVEL, IntToString(nLevel)); + SetScriptParam(DEBUG_PARAM_TARGET, ObjectToString(oTarget)); + ExecuteScript("mydebugscript", oTarget); + return FALSE; + */ + + return TRUE; +} diff --git a/_module/nss/util_c_strftime.nss b/_module/nss/util_c_strftime.nss new file mode 100644 index 0000000..cba8309 --- /dev/null +++ b/_module/nss/util_c_strftime.nss @@ -0,0 +1,97 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_strftime.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Configuration settings for util_i_strftime.nss. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Locale +// ----------------------------------------------------------------------------- +// A locale is a group of localization settings stored as key-value pairs on a +// json object which is then stored on the module and accessed by a name. Some +// functions can take a locale name as an optional parameter so they can access +// those settings. If no name is provided, those functions will use the default +// locale instead. +// ----------------------------------------------------------------------------- + +/// This is the name for the default locale. All settings below will apply to +/// this locale. +const string DEFAULT_LOCALE = "EN_US"; + +// ----------------------------------------------------------------------------- +// Translations +// ----------------------------------------------------------------------------- + +/// This is a 12-element comma-separated list of month names. `%B` evaluates to +/// the item at index `(month - 1) % 12`. +const string DEFAULT_MONTHS = "January, February, March, April, May, June, July, August, September, October, November, December"; + +/// This is a 12-element comma-separated list of abbreviated month names. `%b` +/// evaluates to the item at index `(month - 1) % 12`. +const string DEFAULT_MONTHS_ABBR = "Jan, Feb, Mar, Apr, May, Jun, Jul, Aug, Sep, Oct, Nov, Dec"; + +/// This is a 7-element comma-separated list of weekday names. `%A` evaluates to +/// the item at index `(day - 1) % 7`. +const string DEFAULT_DAYS = "Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday"; + +/// This is a 7-element comma-separated list of abbreviated day names. `%a` +/// evaluates to the item at index `(day - 1) % 7`. +const string DEFAULT_DAYS_ABBR = "Mon, Tue, Wed, Thu, Fri, Sat, Sun"; + +/// This is a 2-element comma-separated list with the preferred representation +/// of AM/PM. Noon is treated as PM and midnight is treated as AM. Evaluated by +/// `%p` (uppercase) and `%P` (lowercase). +const string DEFAULT_AMPM = "AM, PM"; + +/// This is a comma-separated list of suffixes for ordinal numbers. The list +/// should start with the suffix for 0. When formatting using the ordinal flag +/// (e.g., "Today is the %Od day of %B"), the number being formatted is used as +/// an index into this list. If the last two digits of the number are greater +/// than or equal to the length of the list, only the last digit of the number +/// is used. The default value will handle all integers in English. +const string DEFAULT_ORDINAL_SUFFIXES = "th, st, nd, rd, th, th, th, th, th, th, th, th, th, th"; +// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 + +// ----------------------------------------------------------------------------- +// Formatting +// ----------------------------------------------------------------------------- +// These are strings that are used to format dates and times. Refer to the +// comments in `util_i_strftime.nss` for the meaning of format codes. Some codes +// are aliases for these values, so take care to avoid using those codes in +// these values to prevent an infinite loop. +// ----------------------------------------------------------------------------- + +/// This is a string used to format a date and time. Aliased by `%c`. +const string DEFAULT_DATETIME_FORMAT = "%Y-%m-%d %H:%M:%S:%f"; + +/// This is a string used to format a date without the time. Aliased by `%x`. +const string DEFAULT_DATE_FORMAT = "%Y-%m-%d"; + +/// This is a string used to format a time without the date. Aliased by `%X`. +const string DEFAULT_TIME_FORMAT = "%H:%M:%S"; + +/// This is a string used to format a time using AM/PM. Aliased by `%r`. +const string DEFAULT_AMPM_FORMAT = "%I:%M:%S %p"; + +/// This is a string used to format a date and time when era-based formatting is +/// used. If "", will fall back to DEFAULT_DATETIME_FORMAT. Aliased by `%Ec`. +const string DEFAULT_ERA_DATETIME_FORMAT = ""; + +/// This is a string used to format a date without the time when era-based +/// formatting is used. If "", will fall back to DEFAULT_DATE_FORMAT. Aliased by +/// `%Ex`. +const string DEFAULT_ERA_DATE_FORMAT = ""; + +/// This is a string used to format a time without the date when era-based +/// formatting is used. If "", will fall back to DEFAULT_TIME_FORMAT. Aliased by +/// `%EX`. +const string DEFAULT_ERA_TIME_FORMAT = ""; + +/// This is a string used to format years when era-based formatting is used. If +/// "", will always use the current year. Aliased by `%EY`. +const string DEFAULT_ERA_YEAR_FORMAT = "%Ey %EC"; + +/// This is a string used to format the era name when era-based formatting is +/// used. Normally, each era has its own name, but setting this can allow you +/// to display an era name even if you don't set up any eras for your locale. +const string DEFAULT_ERA_NAME = ""; diff --git a/_module/nss/util_c_targeting.nss b/_module/nss/util_c_targeting.nss new file mode 100644 index 0000000..3d5fa01 --- /dev/null +++ b/_module/nss/util_c_targeting.nss @@ -0,0 +1,23 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_targeting.nss +/// @author Ed Burke (tinygiant98) +/// @brief Configuration settings for util_i_targeting.nss. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Targeting Mode Script Handler +// ----------------------------------------------------------------------------- +// You may alter the contents of this function, but do not alter its signature. +// ----------------------------------------------------------------------------- + +/// @brief Custom handler to run scripts associated with targeting hooks. +/// @param sScript The script assigned to the current targeting hook. +/// @param oSelf The PC object assigned to the current targeting event. +void RunTargetingHookScript(string sScript, object oSelf = OBJECT_SELF) +{ + /* Use this function to implement your module's methodology for + running scripts. + + ExecuteScript(sScript, oSelf); + */ +} diff --git a/_module/nss/util_c_unittest.nss b/_module/nss/util_c_unittest.nss new file mode 100644 index 0000000..71099d5 --- /dev/null +++ b/_module/nss/util_c_unittest.nss @@ -0,0 +1,105 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_unittest.nss +/// @author Ed Burke (tinygiant98) +/// @brief Configuration file for util_i_unittest.nss. +/// ---------------------------------------------------------------------------- + +#include "util_i_debug" + +// ----------------------------------------------------------------------------- +// Unit Test Configuration Settings +// ----------------------------------------------------------------------------- + +// Set this value to the color the test title text will be colored to. The value +// can be a value from util_c_color or any other hex value representing a +// color. +// Example Output: Test My Variable Test +// ^^^^ This portion of the text will be affected +const int UNITTEST_TITLE_COLOR = COLOR_CYAN; + +// Set this value to the color the test name text will be colored to. The value +// can be a value from util_c_color or any other hex value representing a +// color. +// Example Output: Test My Variable Test | PASS +// ^^^^^^^^^^^^^^^^ This portion of the text will be affected +const int UNITTEST_NAME_COLOR = COLOR_ORANGE_LIGHT; + +// Set this value to the color the test parameter text will be colored to. The +// value can be a value from util_c_color or any other hex value representing a +// color. +// Example Output: Input: my_input +// Expected: my_assertion +// Received: my_output +// ^^^^^^^^^ This portion of the text will be affected +const int UNITTEST_PARAMETER_COLOR = COLOR_WHITE; + +// Set this value to the color the test parameter text will be colored to. The +// value can be a value from util_c_color or any other hex value representing a +// color. +// Example Output: Input: my_input +// Expected: my_assertion +// ^^^^^^^^^^^^ This portion of the text will be affected +const int UNITTEST_PARAMETER_INPUT = COLOR_GREEN_SEA; + +// Set this value to the color the test parameter text will be colored to. The +// value can be a value from util_c_color or any other hex value representing a +// color. +// Example Output: Received: my_output +// ^^^^^^^^^ This portion of the text will be affected +const int UNITTEST_PARAMETER_RECEIVED = COLOR_PINK; + +// Set this value to the name of the script or event to run in case of a unit +// test failure. +const string UNITTEST_FAILURE_SCRIPT = ""; + +// This value determines whether test results are expanded. Set to TRUE to force +// all test results to show expanded data. Set to FALSE to show expanded data +// only on test failures. +const int UNITTEST_ALWAYS_EXPAND = FALSE; + +// ----------------------------------------------------------------------------- +// Helper Constants and Functions +// ----------------------------------------------------------------------------- +// If you need custom constants, functions, or #includes, you may add them here. +// Since this file is included by util_i_unittest.nss, you do not need to +// #include it to use its constants. +// ----------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Unit Test Output Handler +// ----------------------------------------------------------------------------- +// You may alter the contents of this function, but do not alter its signature. +// ----------------------------------------------------------------------------- + +/// @brief Custom handler to handle reporting unit test results. +/// @param sOutput The formatted and colored output results of a unit test. +void HandleUnitTestOutput(string sOutput) +{ + // This handler can be used to report the unit test output using any module + // debugging or other reporting system. + /* + SendMessageToPC(GetFirstPC(), sOutput); + */ + + Notice(sOutput); +} + +// ----------------------------------------------------------------------------- +// Unit Test Failure Reporting Handler +// ----------------------------------------------------------------------------- +// You may alter the contents of this function, but do not alter its signature. +// ----------------------------------------------------------------------------- + +/// @brief Custom handler to report unit testing failures. +/// @param sOutput The formatted and colored output results of a unit test. +void HandleUnitTestFailure(string sOutput) +{ + // This handler can be used to report unit test failures to module systems + // or take specific action based on a failure. This function will + // generally not be used in a test environment, but may be useful for + // reporting failures in a production environment if unit tests are run + // during module startup. + + if (UNITTEST_FAILURE_SCRIPT != "") + ExecuteScript(UNITTEST_FAILURE_SCRIPT, GetModule()); +} diff --git a/_module/nss/util_c_variables.nss b/_module/nss/util_c_variables.nss new file mode 100644 index 0000000..828c05e --- /dev/null +++ b/_module/nss/util_c_variables.nss @@ -0,0 +1,91 @@ +/// ---------------------------------------------------------------------------- +/// @file util_c_variables.nss +/// @author Ed Burke (tinygiant98) +/// @brief Configuration file for util_i_variables.nss. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Configuration +// ----------------------------------------------------------------------------- + +// This volatile table will be created on the GetModule() object the first time +// a module variable is set. +const string VARIABLE_TABLE_MODULE = "module_variables"; + +// This persitent table will be created on the PC object the first time a player +// variable is set. This table will be stored in the player's .bic file. +const string VARIABLE_TABLE_PC = "player_variables"; + +// A persistent table will be created in a campaign database with the following +// name. The table name will be VARIABLE_TABLE_MODULE above. +const string VARIABLE_CAMPAIGN_DATABASE = "module_variables"; + +// ----------------------------------------------------------------------------- +// Local VarName Constructor +// ----------------------------------------------------------------------------- +// This function is called when attempting to copy variables from a database +// to a game object. Since game objects do not accept additonal fields, such +// as a tag or timestamp, this function is provided to allow construction of +// a unique varname, if desired, from the fields in the database record. You +// may alter the contents of this function, but do not alter its signature. +// ----------------------------------------------------------------------------- + +/// @brief Constructs a varname for a local variable copied from a database. +/// @param oDatabase The database object the variable is sourced from. Will +/// be either a player object, DB_MODULE or DM_CAMPAIGN. +/// @param oTarget The game object the variable will be copied to. +/// @param sVarName VarName field retrieved from database. +/// @param sTag Tag field retrieved from database. +/// @param nType Type field retrieved from database. VARIABLE_TYPE_*, but +/// limited to VARIABLE_TYPE_INT|FLOAT|STRING|OBJECT|LOCATION|JSON. +/// @returns The constructed string that will be used as the varname once +/// copied to the target game object. +string DatabaseToObjectVarName(object oDatabase, object oTarget, string sVarName, + string sTag, int nType) +{ + return sVarName; +} + +// ----------------------------------------------------------------------------- +// Database VarName and Tag Constructors +// ----------------------------------------------------------------------------- +// These functions are called when attempting to copy variables from a game +// object to a database. These functions are provided to allow construction +// of unique varnames and tag from local variables varnames. If the function +// `DatabaseToObjectVarName()` above is used to copy database variables to a +// local object, these functions can be used to reverse the process if +// previously copied variables are returned to a database. You may alter the +// contents of these functions, but do not alter their signatures. +// ----------------------------------------------------------------------------- + +/// @brief Constructs a varname for a local variable copied to a database. +/// @param oSource The game object the variable will be copied from. +/// @param oDatabase The database object the variable will be copied to. Will +/// be either a player object, DB_MODULE or DM_CAMPAIGN. +/// @param sVarName VarName field retrieved from the local variable. +/// @param nType Type field retrieved from database. VARIABLE_TYPE_*, but +/// limited to VARIABLE_TYPE_INT|FLOAT|STRING|OBJECT|LOCATION|JSON. +/// @param sTag sTag as passed to `CopyLocalVariablesToDatabase()`. +/// @returns The constructed string that will be used as the varname once +/// copied to the target game object. +string ObjectToDatabaseVarName(object oSource, object oDatabase, string sVarName, + int nType, string sTag) +{ + return sVarName; +} + +/// @brief Constructs a varname for a local variable copied to a database. +/// @param oSource The game object the variable will be copied from. +/// @param oDatabase The database object the variable will be copied to. Will +/// be either a player object, DB_MODULE or DM_CAMPAIGN. +/// @param sVarName VarName field retrieved from the local variable. +/// @param nType Type field retrieved from database. VARIABLE_TYPE_*, but +/// limited to VARIABLE_TYPE_INT|FLOAT|STRING|OBJECT|LOCATION|JSON. +/// @param sTag sTag as passed to `CopyLocalVariablesToDatabase()`. +/// @returns The constructed string that will be used as the varname once +/// copied to the target game object. +string ObjectToDatabaseTag(object oSource, object oDatabase, string sVarName, + int nType, string sTag) +{ + return sTag; +} diff --git a/_module/nss/util_i_argstack.nss b/_module/nss/util_i_argstack.nss new file mode 100644 index 0000000..f4f920e --- /dev/null +++ b/_module/nss/util_i_argstack.nss @@ -0,0 +1,385 @@ + +/// ---------------------------------------------------------------------------- +/// @file util_i_argstack.nss +/// @author Ed Burke (tinygiant98) +/// @brief Functions for manipulating an argument stack. +/// @details +/// An argument stack provides a method for library functions to send values +/// to other functions without being able to call them directly. This allows +/// library functions to abstract away the connection layer and frees the +/// builder to design plug-and-play systems that don't break when unrelated +/// systems are removed or replaced. +/// +/// Stacks work on a last in - first out basis and are split by variable type. +/// Popping a value will delete and return the last entered value of the +/// specified type stack. Other variable types will not be affected. +/// +/// ```nwscript +/// PushInt(30); +/// PushInt(40); +/// PushInt(50); +/// PushString("test"); +/// +/// int nPop = PopInt(); // nPop = 50 +/// string sPop = PopString(); // sPop = "test"; +/// ```nwscript +/// ---------------------------------------------------------------------------- + +#include "util_i_varlists" + +const string ARGS_DEFAULT_STACK = "ARGS_DEFAULT_STACK"; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Push as value onto the stack. +/// @param nValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushInt(int nValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +int PopInt(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +int PeekInt(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountIntStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param sValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushString(string sValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +string PopString(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +string PeekString(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountStringStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param fValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushFloat(float fValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +float PopFloat(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +float PeekFloat(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountFloatStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param oValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushObject(object oValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +object PopObject(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +object PeekObject(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountObjectStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param lValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to.getlistfloat +/// @returns Count of values on the stack. +int PushLocation(location lValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +location PopLocation(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +location PeekLocation(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountLocationStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param vValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushVector(vector vValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +vector PopVector(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +vector PeekVector(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountVectorStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Push as value onto the stack. +/// @param jValue Value to add to stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Count of values on the stack. +int PushJson(json jValue, string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Pop a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +json PopJson(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Peek a value from the stack. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns Most recent value pushed on the stack. +json PeekJson(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Retrieve the stack size. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @returns The number of values in the stack. +int CountJsonStack(string sListName = "", object oTarget = OBJECT_INVALID); + +/// @brief Clear all stack values. +/// @param sListName [Optional] Name of stack. +/// @param oTarget [Optional] Object stack will be saved to. +/// @note Use this function to ensure all stack values are cleared. +void ClearStacks(string sListName = "", object oTarget = OBJECT_INVALID); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +string _GetListName(string s) +{ + return s == "" ? ARGS_DEFAULT_STACK : s; +} + +object _GetTarget(object o) +{ + if (o == OBJECT_INVALID || GetIsObjectValid(o) == FALSE) + return GetModule(); + return o; +} + +int PushInt(int nValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListInt(_GetTarget(oTarget), 0, nValue, _GetListName(sListName)); +} + +int PopInt(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListInt(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PeekInt(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListInt(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountIntStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountIntList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushString(string sValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListString(_GetTarget(oTarget), 0, sValue, _GetListName(sListName)); +} + +string PopString(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListString(_GetTarget(oTarget), _GetListName(sListName)); +} + +string PeekString(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListString(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountStringStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountStringList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushFloat(float fValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListFloat(_GetTarget(oTarget), 0, fValue, _GetListName(sListName), FALSE); +} + +float PopFloat(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListFloat(_GetTarget(oTarget), _GetListName(sListName)); +} + +float PeekFloat(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListFloat(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountFloatStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountFloatList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushObject(object oValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListObject(_GetTarget(oTarget), 0, oValue, _GetListName(sListName)); +} + +object PopObject(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListObject(_GetTarget(oTarget), _GetListName(sListName)); +} + +object PeekObject(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListObject(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountObjectStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountObjectList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushLocation(location lValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListLocation(_GetTarget(oTarget), 0, lValue, _GetListName(sListName)); +} + +location PopLocation(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListLocation(_GetTarget(oTarget), _GetListName(sListName)); +} + +location PeekLocation(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListLocation(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountLocationStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountLocationList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushVector(vector vValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListVector(_GetTarget(oTarget), 0, vValue, _GetListName(sListName)); +} + +vector PopVector(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListVector(_GetTarget(oTarget), _GetListName(sListName)); +} + +vector PeekVector(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListVector(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountVectorStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountVectorList(_GetTarget(oTarget), _GetListName(sListName)); +} + +int PushJson(json jValue, string sListName = "", object oTarget = OBJECT_INVALID) +{ + return InsertListJson(_GetTarget(oTarget), 0, jValue, _GetListName(sListName)); +} + +json PopJson(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return PopListJson(_GetTarget(oTarget), _GetListName(sListName)); +} + +json PeekJson(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return GetListJson(_GetTarget(oTarget), 0, _GetListName(sListName)); +} + +int CountJsonStack(string sListName = "", object oTarget = OBJECT_INVALID) +{ + return CountJsonList(_GetTarget(oTarget), _GetListName(sListName)); +} + +void ClearStacks(string sListName = "", object oTarget = OBJECT_INVALID) +{ + sListName = _GetListName(sListName); + oTarget = _GetTarget(oTarget); + + DeleteIntList(oTarget, sListName); + DeleteStringList(oTarget, sListName); + DeleteFloatList(oTarget, sListName); + DeleteObjectList(oTarget, sListName); + DeleteLocationList(oTarget, sListName); + DeleteVectorList(oTarget, sListName); + DeleteJsonList(oTarget, sListName); +} diff --git a/_module/nss/util_i_color.nss b/_module/nss/util_i_color.nss new file mode 100644 index 0000000..814f1dc --- /dev/null +++ b/_module/nss/util_i_color.nss @@ -0,0 +1,348 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_color.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Functions to handle colors. +/// @details +/// NWN normally uses color codes to color strings. These codes take the format +/// ``, where RGB are ALT codes (0-0255) for colors. +/// +/// Because color codes are arcane and can't be easily looked up, the functions +/// in this file prefer to use hex color codes. These codes are the same as +/// you'd use in web design and many other areas, so they are easy to look up +/// and can be copied and pasted into other programs. util_c_color.nss provides +/// some hex codes for common uses. +/// +/// This file also contains functions to represent colors as RGB or HSV +/// triplets. HSV (Hue, Saturation, Value) may be particularly useful if you +/// want to play around with shifting colors. +/// +/// ## Acknowledgements +/// - `GetColorCode()` function by rdjparadis. +/// - RGB <-> HSV colors adapted from NWShacker's Named Color Token System. +/// ---------------------------------------------------------------------------- + +#include "x3_inc_string" +#include "util_i_math" +#include "util_c_color" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +/// Used to generate colors from RGB values. NEVER modify this string. +/// @see https://nwn.wiki/display/NWN1/Colour+Tokens +/// @note First character is "nearest to 00" since we can't use `\x00` itself +/// @note COLOR_TOKEN originally by rdjparadis. Converted to NWN:EE escaped +/// characters by Jasperre. +const string COLOR_TOKEN = "\x01\x01\x02\x03\x04\x05\x06\x07\x08\x09\x0A\x0B\x0C\x0D\x0E\x0F\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1A\x1B\x1C\x1D\x1E\x1F\x20\x21\x22\x23\x24\x25\x26\x27\x28\x29\x2A\x2B\x2C\x2D\x2E\x2F\x30\x31\x32\x33\x34\x35\x36\x37\x38\x39\x3A\x3B\x3C\x3D\x3E\x3F\x40\x41\x42\x43\x44\x45\x46\x47\x48\x49\x4A\x4B\x4C\x4D\x4E\x4F\x50\x51\x52\x53\x54\x55\x56\x57\x58\x59\x5A\x5B\x5C\x5D\x5E\x5F\x60\x61\x62\x63\x64\x65\x66\x67\x68\x69\x6A\x6B\x6C\x6D\x6E\x6F\x70\x71\x72\x73\x74\x75\x76\x77\x78\x79\x7A\x7B\x7C\x7D\x7E\x7F\x80\x81\x82\x83\x84\x85\x86\x87\x88\x89\x8A\x8B\x8C\x8D\x8E\x8F\x90\x91\x92\x93\x94\x95\x96\x97\x98\x99\x9A\x9B\x9C\x9D\x9E\x9F\xA0\xA1\xA2\xA3\xA4\xA5\xA6\xA7\xA8\xA9\xAA\xAB\xAC\xAD\xAE\xAF\xB0\xB1\xB2\xB3\xB4\xB5\xB6\xB7\xB8\xB9\xBA\xBB\xBC\xBD\xBE\xBF\xC0\xC1\xC2\xC3\xC4\xC5\xC6\xC7\xC8\xC9\xCA\xCB\xCC\xCD\xCE\xCF\xD0\xD1\xD2\xD3\xD4\xD5\xD6\xD7\xD8\xD9\xDA\xDB\xDC\xDD\xDE\xDF\xE0\xE1\xE2\xE3\xE4\xE5\xE6\xE7\xE8\xE9\xEA\xEB\xEC\xED\xEE\xEF\xF0\xF1\xF2\xF3\xF4\xF5\xF6\xF7\xF8\xF9\xFA\xFB\xFC\xFD\xFE\xFF"; + +// ----------------------------------------------------------------------------- +// Types +// ----------------------------------------------------------------------------- + +struct RGB +{ + int r; + int g; + int b; +}; + +struct HSV +{ + float h; + float s; + float v; +}; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +// ----- Type Creation --------------------------------------------------------- + +/// @brief Create an RBG color struct. +/// @param nRed The value of the red channel (0..255). +/// @param nGreen The value of the green channel (0..255). +/// @param nBlue The value of the blue channel (0..255). +struct RGB GetRGB(int nRed, int nGreen, int nBlue); + +/// @brief Create an HSV color struct. +/// @details The ranges are as follows: +/// 0.0 <= H < 360.0 +/// 0.0 <= S <= 1.0 +/// 0.0 <= V <= 1.0 +/// @param fHue The hue (i.e. location on color wheel). +/// @param fSaturation The saturation (i.e., distance from white/black). +/// @param fValue The value (i.e., brightness of color). +struct HSV GetHSV(float fHue, float fSaturation, float fValue); + +// ----- Type Conversion ------------------------------------------------------- + +/// @brief Convert a hexadecimal color to an RGB struct. +/// @param nColor Hexadecimal to convert to RGB. +struct RGB HexToRGB(int nColor); + +/// @brief Convert an RGB struct to a hexadecimal color. +/// @param rgb RGB to convert to hexadecimal. +int RGBToHex(struct RGB rgb); + +/// @brief Convert an RGB struct to an HSV struct. +/// @param rgb RGB to convert to HSV. +struct HSV RGBToHSV(struct RGB rgb); + +/// @brief Convert an HSV struct to an RGB struct. +/// @param hsv HSV to convert to RGB. +struct RGB HSVToRGB(struct HSV hsv); + +/// @brief Converts a hexadecimal color to an HSV struct. +/// @param nColor Hexadecimal to convert to HSV. +struct HSV HexToHSV(int nColor); + +/// @brief Converts an HSV struct to a hexadecial color. +/// @param hsv HSV to convert to hexadecimal. +int HSVToHex(struct HSV hsv); + +// ----- Coloring Functions ---------------------------------------------------- + +/// @brief Construct a color code that can be used to color a string. +/// @param nRed The intensity of the red channel (0..255). +/// @param nGreen The intensity of the green channel (0..255). +/// @param nBlue The intensity of the blue channel (0..255). +/// @returns A string color code in `` form. +string GetColorCode(int nRed, int nGreen, int nBlue); + +/// @brief Convert a hexadecimal color to a color code. +/// @param nColor Hexadecimal representation of an RGB color. +/// @returns A string color code in `` form. +string HexToColor(int nColor); + +/// @brief Convert a color code prefix to a hexadecimal +/// @param sColor A string color code in `` form. +int ColorToHex(string sColor); + +/// @brief Color a string with a color code. +/// @param sString The string to color. +/// @param sColor A string color code in `` form. +string ColorString(string sString, string sColor); + +/// @brief Color a string with a hexadecimal color. +/// @param sString The string to color. +/// @param nColor A hexadecimal color. +string HexColorString(string sString, int nColor); + +/// @brief Color a string with an RGB color. +/// @param sString The string to color. +/// @param rgb The RGB color struct. +string RGBColorString(string sString, struct RGB rgb); + +/// @brief Color a string with a struct HSV color +/// @param sString The string to color. +/// @param hsv The HSV color struct. +string HSVColorString(string sString, struct HSV hsv); + +/// @brief Remove color codes from a string. +/// @param sString The string to uncolor. +/// @returns sString with color codes removed. +string UnColorString(string sString); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +// ----- Type Creation --------------------------------------------------------- + +struct RGB GetRGB(int nRed, int nGreen, int nBlue) +{ + struct RGB rgb; + + rgb.r = clamp(nRed, 0, 255); + rgb.g = clamp(nGreen, 0, 255); + rgb.b = clamp(nBlue, 0, 255); + + return rgb; +} + +struct HSV GetHSV(float fHue, float fSat, float fVal) +{ + struct HSV hsv; + + hsv.h = fclamp(fHue, 0.0, 360.0); + hsv.s = fclamp(fSat, 0.0, 1.0); + hsv.v = fclamp(fVal, 0.0, 1.0); + + if (hsv.h == 360.0) + hsv.h = 0.0; + + return hsv; +} + +// ----- Type Conversion ------------------------------------------------------- + +struct RGB HexToRGB(int nColor) +{ + int nRed = (nColor & 0xff0000) >> 16; + int nGreen = (nColor & 0x00ff00) >> 8; + int nBlue = (nColor & 0x0000ff); + return GetRGB(nRed, nGreen, nBlue); +} + +int RGBToHex(struct RGB rgb) +{ + int nRed = (clamp(rgb.r, 0, 255) << 16); + int nGreen = (clamp(rgb.g, 0, 255) << 8); + int nBlue = clamp(rgb.b, 0, 255); + return nRed + nGreen + nBlue; +} + +struct HSV RGBToHSV(struct RGB rgb) +{ + // Ensure the RGB values are within defined limits + rgb = GetRGB(rgb.r, rgb.g, rgb.b); + + struct HSV hsv; + + // Convert RGB to a range from 0 - 1 + float fRed = IntToFloat(rgb.r) / 255.0; + float fGreen = IntToFloat(rgb.g) / 255.0; + float fBlue = IntToFloat(rgb.b) / 255.0; + + float fMax = fmax(fRed, fmax(fGreen, fBlue)); + float fMin = fmin(fRed, fmin(fGreen, fBlue)); + float fChroma = fMax - fMin; + + if (fMax > fMin) + { + if (fMax == fRed) + hsv.h = 60.0 * ((fGreen - fBlue) / fChroma); + else if (fMax == fGreen) + hsv.h = 60.0 * ((fBlue - fRed) / fChroma + 2.0); + else + hsv.h = 60.0 * ((fRed - fGreen) / fChroma + 4.0); + + if (hsv.h < 0.0) + hsv.h += 360.0; + } + + if (fMax > 0.0) + hsv.s = fChroma / fMax; + + hsv.v = fMax; + return hsv; +} + +struct RGB HSVToRGB(struct HSV hsv) +{ + // Ensure the HSV values are within defined limits + hsv = GetHSV(hsv.h, hsv.s, hsv.v); + + struct RGB rgb; + + // If value is 0, the resulting color will always be black + if (hsv.v == 0.0) + return rgb; + + // If the saturation is 0, the resulting color will be a shade of grey + if (hsv.s == 0.0) + { + // Scale from white to black based on value + int nValue = FloatToInt(hsv.v * 255.0); + return GetRGB(nValue, nValue, nValue); + } + + float h = hsv.h / 60.0; + float f = frac(h); + int v = FloatToInt(hsv.v * 255.0); + int p = FloatToInt(v * (1.0 - hsv.s)); + int q = FloatToInt(v * (1.0 - hsv.s * f)); + int t = FloatToInt(v * (1.0 - hsv.s * (1.0 - f))); + int i = FloatToInt(h); + + switch (i % 6) + { + case 0: rgb = GetRGB(v, t, p); break; + case 1: rgb = GetRGB(q, v, p); break; + case 2: rgb = GetRGB(p, v, t); break; + case 3: rgb = GetRGB(p, q, v); break; + case 4: rgb = GetRGB(t, p, v); break; + case 5: rgb = GetRGB(v, p, q); break; + } + + return rgb; +} + +struct HSV HexToHSV(int nColor) +{ + return RGBToHSV(HexToRGB(nColor)); +} + +int HSVToHex(struct HSV hsv) +{ + return RGBToHex(HSVToRGB(hsv)); +} + +// ----- Coloring Functions ---------------------------------------------------- + +string GetColorCode(int nRed, int nGreen, int nBlue) +{ + return ""; +} + +string HexToColor(int nColor) +{ + if (nColor < 0 || nColor > 0xffffff) + return ""; + + int nRed = (nColor & 0xff0000) >> 16; + int nGreen = (nColor & 0x00ff00) >> 8; + int nBlue = (nColor & 0x0000ff); + return GetColorCode(nRed, nGreen, nBlue); +} + +int ColorToHex(string sColor) +{ + if (sColor == "") + return -1; + + string sRed = GetSubString(sColor, 2, 1); + string sGreen = GetSubString(sColor, 3, 1); + string sBlue = GetSubString(sColor, 4, 1); + + int nRed = FindSubString(COLOR_TOKEN, sRed) << 16; + int nGreen = FindSubString(COLOR_TOKEN, sGreen) << 8; + int nBlue = FindSubString(COLOR_TOKEN, sBlue); + + return nRed + nGreen + nBlue; +} + +string ColorString(string sString, string sColor) +{ + if (sColor != "") + sString = sColor + sString + ""; + + return sString; +} + +string HexColorString(string sString, int nColor) +{ + string sColor = HexToColor(nColor); + return ColorString(sString, sColor); +} + +string RGBColorString(string sString, struct RGB rgb) +{ + string sColor = GetColorCode(rgb.r, rgb.g, rgb.b); + return ColorString(sString, sColor); +} + +string HSVColorString(string sString, struct HSV hsv) +{ + struct RGB rgb = HSVToRGB(hsv); + return RGBColorString(sString, rgb); +} + +string UnColorString(string sString) +{ + return RegExpReplace("|<\\/c>", sString, ""); +} diff --git a/_module/nss/util_i_constants.nss b/_module/nss/util_i_constants.nss new file mode 100644 index 0000000..31fd56f --- /dev/null +++ b/_module/nss/util_i_constants.nss @@ -0,0 +1,209 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_constants.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions to retrieve the value of a constant from a script file. +/// @details +/// +/// ## Example Usage +/// +/// To retrieve the value of string constant `MODULE_EVENT_ON_NUI` from the Core +/// Framework file `core_i_constants`: +/// ```nwscript +/// struct CONSTANT c = GetConstantString("MODULE_EVENT_ON_NUI", "core_i_constants"); +/// string sSetting = c.sValue; +/// ``` +/// If successful, `sSetting` will contain the string value "OnNUI". If not +/// successful, `c.bError` will be TRUE, `c.sError` will contain the reason for +/// the error, and `c.sValue` will be set to an empty string (""). +/// +/// To retrieve the value of integer constant `EVENT_STATE_OK`from the Core +/// Framework file `core_i_constants`: +/// ```nwscript +/// struct CONSTANT c = GetConstantInt("EVENT_STATE_OK", "core_i_constants"); +/// int nState = c.bError ? -1 : c.nValue; +/// +/// // or... +/// if (!c.bError) +/// { +/// int nState = c.nValue; +/// ... +/// } +/// ``` +/// If successful, `nState` will contain the integer value 0. Since an error +/// value will also return 0, scripts should check `[struct].bError` before +/// using any constant that could return 0 as a valid value. +/// +/// @note These functions require uncompiled `.nss` files, otherwise only base +/// nwscript constants will be retrievable. If you use a tool such as nasher +/// to build your module, ensure you do not filter out the `.nss` files when +/// building. +/// +/// @note Based on clippy's code at +/// https://github.com/Finaldeath/nwscript_utility_scripts +/// ---------------------------------------------------------------------------- + +#include "util_i_debug" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string CONSTANTS_RESULT = "CONSTANTS_RESULT"; +const string CONSTANTS_ERROR_FILE_NOT_FOUND = "FILE NOT FOUND"; +const string CONSTANTS_ERROR_CONSTANT_NOT_FOUND = "VARIABLE DEFINED WITHOUT TYPE"; + +// ----------------------------------------------------------------------------- +// Types +// ----------------------------------------------------------------------------- + +struct CONSTANT +{ + int bError; + string sError; + + string sValue; + int nValue; + float fValue; + + string sFile; + string sConstant; +}; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Retrieves a constant string value from a script file +/// @param sConstant Name of the constant, must match case +/// @param sFile Optional: file to retrieve value from; if omitted, nwscript +/// is assumed +/// @returns a CONSTANT structure containing the following: +/// bError - TRUE if the constant could not be found +/// sError - The reason for the error, if any +/// sValue - The value of the constant retrieved, if successful, or "" +struct CONSTANT GetConstantString(string sConstant, string sFile = ""); + +/// @brief Retrieves a constant integer value from a script file +/// @param sConstant Name of the constant, must match case +/// @param sFile Optional: file to retrieve value from; if omitted, nwscript +/// is assumed +/// @returns a CONSTANT structure containing the following: +/// bError - TRUE if the constant could not be found +/// sError - The reason for the error, if any +/// nValue - The value of the constant retrieved, if successful, or 0 +struct CONSTANT GetConstantInt(string sConstant, string sFile = ""); + +/// @brief Retrieves a constant float value from a script file +/// @param sConstant Name of the constant, must match case +/// @param sFile Optional: file to retrieve value from; if omitted, nwscript +/// is assumed +/// @returns a CONSTANT structure containing the following: +/// bError - TRUE if the constant could not be found +/// sError - The reason for the error, if any +/// fValue - The value of the constant retrieved, if successful, or 0.0 +struct CONSTANT GetConstantFloat(string sConstant, string sFile = ""); + +/// @brief Find an constant name given the constant value. +/// @param sPrefix The prefix portion of the constant name being sought. +/// @param jValue The value of the sPrefix_* constant being sought. This must be +/// a json value to simplify argument passage. Create via a Json* function, +/// such as JsonInt(n), JsonString(s) or JsonFloat(f). +/// @param bSuffixOnly If TRUE, will only return the portion of the constant name +/// found after sPrefix, not including an intervening underscore. +/// @param sFile If passed, sFile will be searched for the appropriate constant name. +/// If not passed, `nwscript.nss` will be searched. +/// @note Does not work with nwscript TRUE/FALSE. Floats that are affected by +/// floating point error, such as 1.67, will also fail to find the correct +/// constant name. Floats that end in .0, such as for DIRECTION_, work correctly. +/// @warning This function is primarily designed for debugging messages. Using it +/// regularly can result in degraded performance. +string GetConstantName(string sPrefix, json jValue, int bSuffixOnly = FALSE, string sFile = ""); + +// ----------------------------------------------------------------------------- +// Private Functions +// ----------------------------------------------------------------------------- + +// Attempts to retrieve the value of sConstant from sFile. If found, the +// appropriate fields in struct CONSTANT are populated. If not, [struct].bError is +// set to TRUE and the reason for failure is populated into [struct].sError. If the +// error cannot be determined, the error returned by ExecuteScriptChunk is +// populated directly into [struct].sError. +struct CONSTANT constants_RetrieveConstant(string sConstant, string sFile, string sType) +{ + int COLOR_KEY = COLOR_BLUE_LIGHT; + int COLOR_VALUE = COLOR_SALMON; + int COLOR_FAILED = COLOR_MESSAGE_FEEDBACK; + + struct CONSTANT c; + string sError, sChunk = "SetLocal" + sType + "(GetModule(), \"" + + CONSTANTS_RESULT + "\", " + sConstant + ");"; + + c.sConstant = sConstant; + c.sFile = sFile == "" ? "nwscript" : sFile; + + if (sFile != "") + sChunk = "#include \"" + sFile + "\" void main() {" + sChunk + "}"; + + if ((sError = ExecuteScriptChunk(sChunk, GetModule(), sFile == "")) != "") + { + c.bError = TRUE; + + if (FindSubString(sError, CONSTANTS_ERROR_FILE_NOT_FOUND) != -1) + c.sError = "Unable to find file `" + c.sFile + ".nss`"; + else if (FindSubString(sError, CONSTANTS_ERROR_CONSTANT_NOT_FOUND) != -1) + c.sError = "Constant `" + c.sConstant + "` not found in `" + c.sFile + ".nss`"; + else + c.sError = sError; + + string sMessage = "[CONSTANTS] " + HexColorString("Failed", COLOR_FAILED) + " to retrieve constant value" + + "\n " + HexColorString("sConstant", COLOR_KEY) + " " + HexColorString(sConstant, COLOR_VALUE) + + "\n " + HexColorString("sFile", COLOR_KEY) + " " + HexColorString(c.sFile, COLOR_VALUE) + + "\n " + HexColorString("Reason", COLOR_KEY) + " " + HexColorString(c.sError, COLOR_VALUE); + Warning(sMessage); + } + + return c; +} + +// ----------------------------------------------------------------------------- +// Public Function Implementations +// ----------------------------------------------------------------------------- + +struct CONSTANT GetConstantString(string sConstant, string sFile = "") +{ + struct CONSTANT c = constants_RetrieveConstant(sConstant, sFile, "String"); + if (!c.bError) + c.sValue = GetLocalString(GetModule(), CONSTANTS_RESULT); + + return c; +} + +struct CONSTANT GetConstantInt(string sConstant, string sFile = "") +{ + struct CONSTANT c = constants_RetrieveConstant(sConstant, sFile, "Int"); + if (!c.bError) + c.nValue = GetLocalInt(GetModule(), CONSTANTS_RESULT); + + return c; +} + +struct CONSTANT GetConstantFloat(string sConstant, string sFile = "") +{ + struct CONSTANT c = constants_RetrieveConstant(sConstant, sFile, "Float"); + if (!c.bError) + c.fValue = GetLocalFloat(GetModule(), CONSTANTS_RESULT); + + return c; +} + +string GetConstantName(string sPrefix, json jValue, int bSuffixOnly = FALSE, string sFile = "") +{ + if (sFile == "") sFile = "nwscript"; + + sPrefix = GetStringUpperCase(bSuffixOnly ? sPrefix + "_?(" : "(" + sPrefix); + json jMatch = RegExpMatch(sPrefix + ".*?)(?: |=).*?=\\s*(" + + JsonDump(jValue) + ")\\s*;", ResManGetFileContents(sFile, RESTYPE_NSS)); + + return jMatch != JsonArray() ? JsonGetString(JsonArrayGet(jMatch, 1)) : ""; +} diff --git a/_module/nss/util_i_csvlists.nss b/_module/nss/util_i_csvlists.nss new file mode 100644 index 0000000..4f18e51 --- /dev/null +++ b/_module/nss/util_i_csvlists.nss @@ -0,0 +1,353 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_csvlists.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions for manipulating comma-separated value (CSV) lists. +/// @details +/// +/// ## Usage: +/// +/// ```nwscript +/// string sKnight, sKnights = "Lancelot, Galahad, Robin"; +/// int i, nCount = CountList(sKnights); +/// for (i = 0; i < nCount; i++) +/// { +/// sKnight = GetListItem(sKnights, i); +/// SpeakString("Sir " + sKnight); +/// } +/// +/// int bBedivere = HasListItem(sKnights, "Bedivere"); +/// SpeakString("Bedivere " + (bBedivere ? "is" : "is not") + " in the party."); +/// +/// sKnights = AddListItem(sKnights, "Bedivere"); +/// bBedivere = HasListItem(sKnights, "Bedivere"); +/// SpeakString("Bedivere " + (bBedivere ? "is" : "is not") + " in the party."); +/// +/// int nRobin = FindListItem(sKnights, "Robin"); +/// SpeakString("Robin is knight " + IntToString(nRobin) + " in the party."); +/// ``` +/// ---------------------------------------------------------------------------- + +#include "x3_inc_string" +#include "util_i_math" +#include "util_i_strings" + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Trim excess space around commas and, optionally, remove excess commas/ +/// empty list items. +/// @param sList The CSV list to normalize. +/// @param bRemoveEmpty TRUE to remove empty items. +string NormalizeList(string sList, int bRemoveEmpty = TRUE); + +/// @brief Return the number of items in a CSV list. +/// @param sList The CSV list to count. +int CountList(string sList); + +/// @brief Add an item to a CSV list. +/// @param sList The CSV list to add the item to. +/// @param sListItem The item to add to sList. +/// @param bAddUnique If TRUE, will only add the item to the list if it is not +/// already there. +/// @returns A modified copy of sList with sListItem added. +string AddListItem(string sList, string sListItem, int bAddUnique = FALSE); + +/// @brief Insert an item into a CSV list. +/// @param sList The CSV list to insert the item into. +/// @param sListItem The item to insert into sList. +/// @param nIndex The index of the item to insert (0-based). +/// @param bAddUnique If TRUE, will only insert the item to the list if it is not +/// already there. +/// @returns A modified copy of sList with sListItem inserted. +string InsertListItem(string sList, string sListItem, int nIndex = -1, int bAddUnique = FALSE); + +/// @brief Modify an existing item in a CSV list. +/// @param sList The CSV list to modify. +/// @param sListItem The item to insert at nIndex. +/// @param nIndex The index of the item to modify (0-based). +/// @param bAddUnique If TRUE, will only modify the item to the list if it is not +/// already there. +/// @returns A modified copy of sList with item at nIndex modified. +/// @note If nIndex is out of bounds for sList, no values will be modified. +/// @warning If bAddUnique is TRUE and a non-unique value is set, the value with a lower +/// list index will be kept and values with higher list indices removed. +string SetListItem(string sList, string sListItem, int nIndex = -1, int bAddUnique = FALSE); + +/// @brief Return the item at an index in a CSV list. +/// @param sList The CSV list to get the item from. +/// @param nIndex The index of the item to get (0-based). +string GetListItem(string sList, int nIndex = 0); + +/// @brief Return the index of a value in a CSV list. +/// @param sList The CSV list to search. +/// @param sListItem The value to search for. +/// @param nNth The nth repetition of sListItem. +/// @returns -1 if the item was not found in the list. +int FindListItem(string sList, string sListItem, int nNth = 0); + +/// @brief Return whether a CSV list contains a value. +/// @param sList The CSV list to search. +/// @param sListItem The value to search for. +/// @returns TRUE if the item is in the list, otherwise FALSE. +int HasListItem(string sList, string sListItem); + +/// @brief Delete the item at an index in a CSV list. +/// @param sList The CSV list to delete the item from. +/// @param nIndex The index of the item to delete (0-based). +/// @returns A modified copy of sList with the item deleted. +string DeleteListItem(string sList, int nIndex = 0); + +/// @brief Delete the first occurrence of an item in a CSV list. +/// @param sList The CSV list to remove the item from. +/// @param sListItem The value to remove from the list. +/// @param nNth The nth repetition of sListItem. +/// @returns A modified copy of sList with the item removed. +string RemoveListItem(string sList, string sListItem, int nNth = 0); + +/// @brief Copy items from one CSV list to another. +/// @param sSource The CSV list to copy items from. +/// @param sTarget The CSV list to copy items to. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of items to copy. +/// @param bAddUnique If TRUE, will only copy items to sTarget if they are not +/// already there. +/// @returns A modified copy of sTarget with the items added to the end. +string CopyListItem(string sSource, string sTarget, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Merge the contents of two CSV lists. +/// @param sList1 The first CSV list. +/// @param sList2 The second CSV list. +/// @param bAddUnique If TRUE, will only put items in the returned list if they +/// are not already there. +/// @returns A CSV list containing the items from each list. +string MergeLists(string sList1, string sList2, int bAddUnique = FALSE); + +/// @brief Add an item to a CSV list saved as a local variable on an object. +/// @param oObject The object on which the local variable is saved. +/// @param sListName The varname for the local variable. +/// @param sListItem The item to add to the list. +/// @param bAddUnique If TRUE, will only add the item to the list if it is not +/// already there. +/// @returns The updated copy of the list with sListItem added. +string AddLocalListItem(object oObject, string sListName, string sListItem, int bAddUnique = FALSE); + +/// @brief Delete an item in a CSV list saved as a local variable on an object. +/// @param oObject The object on which the local variable is saved. +/// @param sListName The varname for the local variable. +/// @param nIndex The index of the item to delete (0-based). +/// @returns The updated copy of the list with the item at nIndex deleted. +string DeleteLocalListItem(object oObject, string sListName, int nIndex = 0); + +/// @brief Remove an item in a CSV list saved as a local variable on an object. +/// @param oObject The object on which the local variable is saved. +/// @param sListName The varname for the local variable. +/// @param sListItem The value to remove from the list. +/// @param nNth The nth repetition of sListItem. +/// @returns The updated copy of the list with the first instance of sListItem +/// removed. +string RemoveLocalListItem(object oObject, string sListName, string sListItem, int nNth = 0); + +/// @brief Merge the contents of a CSV list with those of a CSV list stored as a +/// local variable on an object. +/// @param oObject The object on which the local variable is saved. +/// @param sListName The varname for the local variable. +/// @param sListToMerge The CSV list to merge into the saved list. +/// @param bAddUnique If TRUE, will only put items in the returned list if they +/// are not already there. +/// @returns The updated copy of the list with all items from sListToMerge +/// added. +string MergeLocalList(object oObject, string sListName, string sListToMerge, int bAddUnique = FALSE); + +/// @brief Convert a comma-separated value list to a JSON array. +/// @param sList Source CSV list. +/// @param bNormalize TRUE to remove excess spaces and values. See NormalizeList(). +/// @returns JSON array representation of CSV list. +json ListToJson(string sList, int bNormalize = TRUE); + +/// @brief Convert a JSON array to a comma-separate value list. +/// @param jList JSON array list. +/// @param bNormalize TRUE to remove excess spaces and values. See NormalizeList(). +/// @returns CSV list of JSON array values. +string JsonToList(json jList, int bNormalize = TRUE); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +string NormalizeList(string sList, int bRemoveEmpty = TRUE) +{ + string sRegex = "(?:[\\s]*,[\\s]*)" + (bRemoveEmpty ? "+" : ""); + sList = RegExpReplace(sRegex, sList, ","); + return TrimString(bRemoveEmpty ? RegExpReplace("^[\\s]*,|,[\\s]*$", sList, "") : sList); +} + +int CountList(string sList) +{ + if (sList == "") + return 0; + + return GetSubStringCount(sList, ",") + 1; +} + +string AddListItem(string sList, string sListItem, int bAddUnique = FALSE) +{ + sList = NormalizeList(sList); + sListItem = TrimString(sListItem); + + if (bAddUnique && HasListItem(sList, sListItem)) + return sList; + + if (sList != "") + return sList + "," + sListItem; + + return sListItem; +} + +string InsertListItem(string sList, string sListItem, int nIndex = -1, int bAddUnique = FALSE) +{ + if (nIndex == -1 || sList == "" || nIndex > CountList(sList) - 1) + return AddListItem(sList, sListItem, bAddUnique); + + if (nIndex < 0) nIndex = 0; + json jList = JsonArrayInsert(ListToJson(sList), JsonString(sListItem), nIndex); + + if (bAddUnique == TRUE) + jList = JsonArrayTransform(jList, JSON_ARRAY_UNIQUE); + + return JsonToList(jList); +} + +string SetListItem(string sList, string sListItem, int nIndex = -1, int bAddUnique = FALSE) +{ + if (nIndex < 0 || nIndex > (CountList(sList) - 1)) + return sList; + + json jList = JsonArraySet(ListToJson(sList), nIndex, JsonString(sListItem)); + + if (bAddUnique == TRUE) + jList = JsonArrayTransform(jList, JSON_ARRAY_UNIQUE); + + return JsonToList(jList); +} + +string GetListItem(string sList, int nIndex = 0) +{ + if (nIndex < 0 || sList == "" || nIndex > (CountList(sList) - 1)) + return ""; + + return JsonGetString(JsonArrayGet(ListToJson(sList), nIndex)); +} + +int FindListItem(string sList, string sListItem, int nNth = 0) +{ + json jIndex = JsonFind(ListToJson(sList), JsonString(TrimString(sListItem)), nNth); + return jIndex == JSON_NULL ? -1 : JsonGetInt(jIndex); +} + +int HasListItem(string sList, string sListItem) +{ + return (FindListItem(sList, sListItem) > -1); +} + +string DeleteListItem(string sList, int nIndex = 0) +{ + if (nIndex < 0 || sList == "" || nIndex > (CountList(sList) - 1)) + return sList; + + return JsonToList(JsonArrayDel(ListToJson(sList), nIndex)); +} + +string RemoveListItem(string sList, string sListItem, int nNth = 0) +{ + return DeleteListItem(sList, FindListItem(sList, sListItem, nNth)); +} + +string CopyListItem(string sSource, string sTarget, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + int i, nCount = CountList(sSource); + + if (nIndex < 0 || nIndex >= nCount || !nCount) + return sSource; + + nRange = clamp(nRange, 1, nCount - nIndex); + + for (i = 0; i < nRange; i++) + sTarget = AddListItem(sTarget, GetListItem(sSource, nIndex + i), bAddUnique); + + return sTarget; +} + +string MergeLists(string sList1, string sList2, int bAddUnique = FALSE) +{ + if (sList1 != "" && sList2 == "") + return sList1; + else if (sList1 == "" && sList2 != "") + return sList2; + else if (sList1 == "" && sList2 == "") + return ""; + + string sList = sList1 + "," + sList2; + + if (bAddUnique) + sList = JsonToList(JsonArrayTransform(ListToJson(sList), JSON_ARRAY_UNIQUE)); + + return bAddUnique ? sList : NormalizeList(sList); +} + +string AddLocalListItem(object oObject, string sListName, string sListItem, int bAddUnique = FALSE) +{ + string sList = GetLocalString(oObject, sListName); + sList = AddListItem(sList, sListItem, bAddUnique); + SetLocalString(oObject, sListName, sList); + return sList; +} + +string DeleteLocalListItem(object oObject, string sListName, int nIndex = 0) +{ + string sList = GetLocalString(oObject, sListName); + sList = DeleteListItem(sList, nIndex); + SetLocalString(oObject, sListName, sList); + return sList; +} + +string RemoveLocalListItem(object oObject, string sListName, string sListItem, int nNth = 0) +{ + string sList = GetLocalString(oObject, sListName); + sList = RemoveListItem(sList, sListItem, nNth); + SetLocalString(oObject, sListName, sList); + return sList; +} + +string MergeLocalList(object oObject, string sListName, string sListToMerge, int bAddUnique = FALSE) +{ + string sList = GetLocalString(oObject, sListName); + sList = MergeLists(sList, sListToMerge, bAddUnique); + SetLocalString(oObject, sListName, sList); + return sList; +} + +json ListToJson(string sList, int bNormalize = TRUE) +{ + if (sList == "") + return JSON_ARRAY; + + if (bNormalize) + sList = NormalizeList(sList); + + sList = RegExpReplace("\"", sList, "\\\""); + return JsonParse("[\"" + RegExpReplace(",", sList, "\",\"") + "\"]"); +} + +string JsonToList(json jList, int bNormalize = TRUE) +{ + if (JsonGetType(jList) != JSON_TYPE_ARRAY) + return ""; + + string sList; + int n; for (n; n < JsonGetLength(jList); n++) + sList += (sList == "" ? "" : ",") + JsonGetString(JsonArrayGet(jList, n)); + + return bNormalize ? NormalizeList(sList) : sList; +} diff --git a/_module/nss/util_i_datapoint.nss b/_module/nss/util_i_datapoint.nss new file mode 100644 index 0000000..a10e653 --- /dev/null +++ b/_module/nss/util_i_datapoint.nss @@ -0,0 +1,132 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_datapoint.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions for creating and interacting with datapoints, which are +/// invisible objects used to hold variables specific to a system. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string DATA_PREFIX = "Datapoint: "; +const string DATA_POINT = "x1_hen_inv"; ///< Resref for data points +const string DATA_ITEM = "nw_it_msmlmisc22"; ///< Resref for data items + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Creates a datapoint (placeable) that stores variables for a +/// specified system +/// @param sSystem Name of system associated with this datapoint +/// @param oOwner (optional) Parent object of this datapoint; if omitted, +/// defaults to GetModule(); +/// @note A datapoint is created at oOwner's location; if oOwner is invalid or +/// is an area object, the datapoint is created at the module starting +/// location. +/// @returns sSystem's datapoint object +object CreateDatapoint(string sSystem, object oOwner = OBJECT_INVALID); + +/// @brief Retrieves a datapoint (placeable) that stores variables for a +/// specified system +/// @param sSystem Name of system associated with this datapoint +/// @param oOwner (optional) Parent object of this datapoint; if omitted, +/// defaults to GetModule() +/// @param bCreate If TRUE and the datapoint cannot be found, a new datapoint +/// will be created at oOwner's location; if oOwner is invalid or is an +/// area object, the datapoint is created at the module starting location +/// @returns sSystem's datapoint object +object GetDatapoint(string sSystem, object oOwner = OBJECT_INVALID, int bCreate = TRUE); + +/// @brief Sets a datapoint (game object) as the object that stores variables +/// for a specified system +/// @param sSystem Name of system associated with this datapoint +/// @param oTarget Object to be used as a datapoint +/// @param oOwner (optional) Parent object of this datapoint; if omitted, +/// default to GetModule() +/// @note Allows any valid game object to be used as a datapoint +void SetDatapoint(string sSystem, object oTarget, object oOwner = OBJECT_INVALID); + +/// @brief Creates a data item (item) that stores variables for a specified +/// sub-system +/// @param oDatapoint Datapoint object on which to place the data item +/// @param sSubSystem Name of sub-system associated with this data item +/// @returns sSubSystem's data item object +object CreateDataItem(object oDatapoint, string sSubSystem); + +/// @brief Retrieves a data item (item) that stores variables for a specified +/// sub-system +/// @param oDatapoint Datapoint object from which to retrieve the data item +/// @param sSubSystem Name of sub-system associated with the data item +/// @returns sSubSystem's data item object +object GetDataItem(object oDatapoint, string sSubSystem); + +/// @brief Sets a data item (item) as the object that stores variables for a +/// specified sub-system +/// @param oDatapoint Datapoint object on which to place the data item +/// @param sSubSystem Name of sub-system assocaited with the data item +/// @param oItem Item to be used as a data item +/// @note oItem must a valid game item that can be placed into an object's +/// inventory +void SetDataItem(object oDatapoint, string sSubSystem, object oItem); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +object CreateDatapoint(string sSystem, object oOwner = OBJECT_INVALID) +{ + if (oOwner == OBJECT_INVALID) + oOwner = GetModule(); + + location lLoc = GetLocation(oOwner); + if (!GetObjectType(oOwner)) + lLoc = GetStartingLocation(); + + object oData = CreateObject(OBJECT_TYPE_PLACEABLE, DATA_POINT, lLoc); + SetName(oData, DATA_PREFIX + sSystem); + SetUseableFlag(oData, FALSE); + SetDatapoint(sSystem, oData, oOwner); + return oData; +} + +object GetDatapoint(string sSystem, object oOwner = OBJECT_INVALID, int bCreate = TRUE) +{ + if (oOwner == OBJECT_INVALID) + oOwner = GetModule(); + + object oData = GetLocalObject(oOwner, DATA_PREFIX + sSystem); + + if (!GetIsObjectValid(oData) && bCreate) + oData = CreateDatapoint(sSystem, oOwner); + + return oData; +} + +void SetDatapoint(string sSystem, object oTarget, object oOwner = OBJECT_INVALID) +{ + if (oOwner == OBJECT_INVALID) + oOwner = GetModule(); + + SetLocalObject(oOwner, DATA_PREFIX + sSystem, oTarget); +} + +object CreateDataItem(object oDatapoint, string sSubSystem) +{ + object oItem = CreateItemOnObject(DATA_ITEM, oDatapoint); + SetLocalObject(oDatapoint, sSubSystem, oItem); + SetName(oItem, sSubSystem); + return oItem; +} + +object GetDataItem(object oDatapoint, string sSubSystem) +{ + return GetLocalObject(oDatapoint, sSubSystem); +} + +void SetDataItem(object oDatapoint, string sSubSystem, object oItem) +{ + SetLocalObject(oDatapoint, sSubSystem, oItem); +} diff --git a/_module/nss/util_i_debug.nss b/_module/nss/util_i_debug.nss new file mode 100644 index 0000000..01e3aea --- /dev/null +++ b/_module/nss/util_i_debug.nss @@ -0,0 +1,324 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_debug.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions for generating debug messages. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// VarNames +const string DEBUG_COLOR = "DEBUG_COLOR"; +const string DEBUG_LEVEL = "DEBUG_LEVEL"; +const string DEBUG_LOG = "DEBUG_LOG"; +const string DEBUG_OVERRIDE = "DEBUG_OVERRIDE"; +const string DEBUG_PREFIX = "DEBUG_PREFIX"; +const string DEBUG_DISPATCH = "DEBUG_DISPATCH"; + +// Debug levels +const int DEBUG_LEVEL_NONE = 0; ///< No debug level set +const int DEBUG_LEVEL_CRITICAL = 1; ///< Errors severe enough to stop the script +const int DEBUG_LEVEL_ERROR = 2; ///< Indicates the script malfunctioned in some way +const int DEBUG_LEVEL_WARNING = 3; ///< Indicates that unexpected behavior may occur +const int DEBUG_LEVEL_NOTICE = 4; ///< Information to track the flow of the function +const int DEBUG_LEVEL_DEBUG = 5; ///< Data dumps used for debugging + +// Debug logging +const int DEBUG_LOG_NONE = 0x0; ///< Do not log debug messages +const int DEBUG_LOG_FILE = 0x1; ///< Send debug messages to the log file +const int DEBUG_LOG_DM = 0x2; ///< Send debug messages to online DMs +const int DEBUG_LOG_PC = 0x4; ///< Send debug messages to the first PC +const int DEBUG_LOG_LIST = 0x8; ///< Send debug messages to the dispatch list +const int DEBUG_LOG_ALL = 0xf; ///< Send messages to the log file, DMs, and first PC or dispatch list + +#include "util_c_debug" +#include "util_i_color" +#include "util_i_varlists" + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Temporarily override the debug level for all objects. +/// @param nLevel The maximum verbosity of messages to show. Use FALSE to stop +/// overriding the debug level. +void OverrideDebugLevel(int nLevel); + +/// @brief Return the verbosity of debug messages displayed for an object. +/// @param oTarget The object to check the debug level of. If no debug level has +/// been set on oTarget, will use the module instead. +/// @returns A `DEBUG_LEVEL_*` constant representing the maximum verbosity of +/// messages that will be displayed when debugging oTarget. +int GetDebugLevel(object oTarget = OBJECT_SELF); + +/// @brief Set the verbosity of debug messages displayed for an object. +/// @param nLevel A `DEBUG_LEVEL_*` constant representing the maximum verbosity +/// of messages that will be displayed when debugging oTarget. If set to +/// `DEBUG_LEVEL_NONE`, oTarget will use the module's debug level instead. +/// @param oTarget The object to set the debug level of. If no debug level has +/// been set on oTarget, will use the module instead. +void SetDebugLevel(int nLevel, object oTarget = OBJECT_SELF); + +/// @brief Return the color of debug messages of a given level. +/// @param nLevel A `DEBUG_LEVEL_*` constant representing the verbosity of debug +/// messsages to get the color for. +/// @returns A color code (in form). +string GetDebugColor(int nLevel); + +/// @brief Set the color of debug messages of a given level. +/// @param nLevel A `DEBUG_LEVEL_*` constant representing the verbosity of debug +/// messsages to get the color for. +/// @param sColor A color core (in form) for the debug messages. If "", +/// will use the default color code for the level. +void SetDebugColor(int nLevel, string sColor = ""); + +/// @brief Return the prefix an object uses before its debug messages. +/// @param oTarget The target to check for a prefix. +/// @returns The user-defined prefix if one has been set. If it has not, will +/// return the object's tag (or name, if the object has no tag) in square +/// brackets. +string GetDebugPrefix(object oTarget = OBJECT_SELF); + +/// @brief Set the prefix an object uses before its debug messages. +/// @param sPrefix The prefix to set. You can include color codes in the prefix, +/// but you can also set thedefault color code for all prefixes using +/// `SetDebugColor(DEBUG_COLOR_NONE, sColor);`. +/// @param oTarget The target to set the prefix for. +void SetDebugPrefix(string sPrefix, object oTarget = OBJECT_SELF); + +/// @brief Return the enabled debug logging destinations. +/// @returns A bitmask of `DEBUG_LOG_*` values. +int GetDebugLogging(); + +/// @brief Set the enabled debug logging destinations. +/// @param nEnabled A bitmask of `DEBUG_LOG_*` destinations to enable. +void SetDebugLogging(int nEnabled); + +/// @brief Add a player object to the debug message dispatch list. Player +/// objects on the dispatch list will receive debug messages if the +/// module's DEBUG_LOG_LEVEL includes DEBUG_LOG_LIST. +/// @param oPC Player object to add. +void AddDebugLoggingPC(object oPC); + +/// @brief Remove a player object from the debug dispatch list. +/// @param oPC Player object to remove. +void RemoveDebugLoggingPC(object oPC); + +/// @brief Return whether debug messages of a given level will be logged on a +/// target. Useful for avoiding spending cycles computing extra debug +/// information if it will not be shown. +/// @param nLevel A `DEBUG_LEVEL_*` constant representing the message verbosity. +/// @param oTarget The object that would be debugged. +/// @returns TRUE if messages of nLevel would be logged on oTarget; FALSE +/// otherwise. +int IsDebugging(int nLevel, object oTarget = OBJECT_SELF); + +/// If oTarget has a debug level of nLevel or higher, logs sMessages to all +/// destinations set with SetDebugLogging(). If no debug level is set on +/// oTarget, +/// will debug using the module's debug level instead. +/// @brief Display a debug message. +/// @details If the target has a debug level of nLevel or higher, sMessage will +/// be sent to all destinations enabled by SetDebugLogging(). If no debug +/// level is set on oTarget, will debug using the module's debug level +/// instead. +/// @param sMessage The message to display. +/// @param nLevel A `DEBUG_LEVEL_*` constant representing the message verbosity. +/// @param oTarget The object originating the message. +void Debug(string sMessage, int nLevel = DEBUG_LEVEL_DEBUG, object oTarget = OBJECT_SELF); + +/// @brief Display a general notice message. Alias for Debug(). +/// @param sMessage The message to display. +/// @param oTarget The object originating the message. +void Notice(string sMessage, object oTarget = OBJECT_SELF); + +/// @brief Display a warning message. Alias for Debug(). +/// @param sMessage The message to display. +/// @param oTarget The object originating the message. +void Warning(string sMessage, object oTarget = OBJECT_SELF); + +/// @brief Display an error message. Alias for Debug(). +/// @param sMessage The message to display. +/// @param oTarget The object originating the message. +void Error(string sMessage, object oTarget = OBJECT_SELF); + +/// @brief Display a critical error message. Alias for Debug(). +/// @param sMessage The message to display. +/// @param oTarget The object originating the message. +void CriticalError(string sMessage, object oTarget = OBJECT_SELF); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +void OverrideDebugLevel(int nLevel) +{ + nLevel = clamp(nLevel, DEBUG_LEVEL_NONE, DEBUG_LEVEL_DEBUG); + SetLocalInt(GetModule(), DEBUG_OVERRIDE, nLevel); +} + +int GetDebugLevel(object oTarget = OBJECT_SELF) +{ + object oModule = GetModule(); + int nOverride = GetLocalInt(oModule, DEBUG_OVERRIDE); + if (nOverride) + return nOverride; + + int nModule = GetLocalInt(oModule, DEBUG_LEVEL); + if (oTarget == oModule || !GetIsObjectValid(oTarget)) + return nModule; + + int nLevel = GetLocalInt(oTarget, DEBUG_LEVEL); + return (nLevel ? nLevel : nModule ? nModule : DEBUG_LEVEL_CRITICAL); +} + +void SetDebugLevel(int nLevel, object oTarget = OBJECT_SELF) +{ + SetLocalInt(oTarget, DEBUG_LEVEL, nLevel); +} + +string GetDebugColor(int nLevel) +{ + string sColor = GetLocalString(GetModule(), DEBUG_COLOR + IntToString(nLevel)); + + if (sColor == "") + { + int nColor; + switch (nLevel) + { + case DEBUG_LEVEL_CRITICAL: nColor = COLOR_RED; break; + case DEBUG_LEVEL_ERROR: nColor = COLOR_ORANGE_DARK; break; + case DEBUG_LEVEL_WARNING: nColor = COLOR_ORANGE_LIGHT; break; + case DEBUG_LEVEL_NOTICE: nColor = COLOR_YELLOW; break; + case DEBUG_LEVEL_NONE: nColor = COLOR_GREEN_LIGHT; break; + default: nColor = COLOR_GRAY_LIGHT; break; + } + + sColor = HexToColor(nColor); + SetDebugColor(nLevel, sColor); + } + + return sColor; +} + +void SetDebugColor(int nLevel, string sColor = "") +{ + SetLocalString(GetModule(), DEBUG_COLOR + IntToString(nLevel), sColor); +} + +string GetDebugPrefix(object oTarget = OBJECT_SELF) +{ + string sColor = GetDebugColor(DEBUG_LEVEL_NONE); + string sPrefix = GetLocalString(oTarget, DEBUG_PREFIX); + if (sPrefix == "") + { + if (!GetIsObjectValid(oTarget)) + { + sColor = GetDebugColor(DEBUG_LEVEL_WARNING); + sPrefix = "Invalid Object: #" + ObjectToString(oTarget); + } + else + sPrefix = (sPrefix = GetTag(oTarget)) == "" ? GetName(oTarget) : sPrefix; + + sPrefix = "[" + sPrefix + "]"; + } + + return ColorString(sPrefix, sColor); +} + +void SetDebugPrefix(string sPrefix, object oTarget = OBJECT_SELF) +{ + SetLocalString(oTarget, DEBUG_PREFIX, sPrefix); +} + +int GetDebugLogging() +{ + return GetLocalInt(GetModule(), DEBUG_LOG); +} + +void SetDebugLogging(int nEnabled) +{ + SetLocalInt(GetModule(), DEBUG_LOG, nEnabled); +} + +void AddDebugLoggingPC(object oPC) +{ + if (GetIsPC(oPC)) + AddListObject(GetModule(), oPC, DEBUG_DISPATCH, TRUE); +} + +void RemoveDebugLoggingPC(object oPC) +{ + RemoveListObject(GetModule(), oPC, DEBUG_DISPATCH); +} + +int IsDebugging(int nLevel, object oTarget = OBJECT_SELF) +{ + return (nLevel <= GetDebugLevel(oTarget)); +} + +void Debug(string sMessage, int nLevel = DEBUG_LEVEL_DEBUG, object oTarget = OBJECT_SELF) +{ + if (IsDebugging(nLevel, oTarget)) + { + string sColor = GetDebugColor(nLevel); + string sPrefix = GetDebugPrefix(oTarget) + " "; + + switch (nLevel) + { + case DEBUG_LEVEL_CRITICAL: sPrefix += "[Critical Error] "; break; + case DEBUG_LEVEL_ERROR: sPrefix += "[Error] "; break; + case DEBUG_LEVEL_WARNING: sPrefix += "[Warning] "; break; + } + + if (!HandleDebug(sPrefix, sMessage, nLevel, oTarget)) + return; + + sMessage = sPrefix + sMessage; + int nLogging = GetLocalInt(GetModule(), DEBUG_LOG); + + if (nLogging & DEBUG_LOG_FILE) + WriteTimestampedLogEntry(UnColorString(sMessage)); + + sMessage = ColorString(sMessage, sColor); + + if (nLogging & DEBUG_LOG_DM) + SendMessageToAllDMs(sMessage); + + if (nLogging & DEBUG_LOG_PC) + SendMessageToPC(GetFirstPC(), sMessage); + + if (nLogging & DEBUG_LOG_LIST) + { + json jDispatchList = GetObjectList(GetModule(), DEBUG_DISPATCH); + int n; for (n; n < JsonGetLength(jDispatchList); n++) + { + object oPC = GetListObject(GetModule(), n, DEBUG_DISPATCH); + if (GetIsPC(oPC) && !((nLogging & DEBUG_LOG_PC) && oPC == GetFirstPC())) + SendMessageToPC(oPC, sMessage); + } + } + } +} + +void Notice(string sMessage, object oTarget = OBJECT_SELF) +{ + Debug(sMessage, DEBUG_LEVEL_NOTICE, oTarget); +} + +void Warning(string sMessage, object oTarget = OBJECT_SELF) +{ + Debug(sMessage, DEBUG_LEVEL_WARNING, oTarget); +} + +void Error(string sMessage, object oTarget = OBJECT_SELF) +{ + Debug(sMessage, DEBUG_LEVEL_ERROR, oTarget); +} + +void CriticalError(string sMessage, object oTarget = OBJECT_SELF) +{ + Debug(sMessage, DEBUG_LEVEL_CRITICAL, oTarget); +} diff --git a/_module/nss/util_i_libraries.nss b/_module/nss/util_i_libraries.nss new file mode 100644 index 0000000..24b857e --- /dev/null +++ b/_module/nss/util_i_libraries.nss @@ -0,0 +1,481 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_libraries.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief This file holds functions for packing scripts into libraries. This +/// allows the builder to dramatically reduce the module script count by +/// keeping related scripts in the same file. +/// @details +/// Libraries allow the builder to encapsulate many scripts into one, +/// dramatically reducing the script count in the module. In a library, each +/// script is a function bound to a unique name and/or number. When the library +/// is called, the name is routed to the proper function. +/// +/// Since each script defined by a library has a unique name to identify it, the +/// builder can execute a library script without having to know the file it is +/// located in. This makes it easy to create script systems to override behavior +/// of another system; you don't have to edit the other system's code, you just +/// implement your own function to override it. +/// +/// ## Anatomy of a Library +/// This is an example of a simple library: +/// +/// ``` nwscript +/// #include "util_i_libraries" +/// +/// void MyFunction() +/// { +/// // ... +/// } +/// +/// void MyOtherFunction() +/// { +/// // ... +/// } +/// +/// void OnLibraryLoad() +/// { +/// RegisterLibraryScript("MyFunction"); +/// RegisterLibraryScript("MyOtherFunction"); +/// } +/// ``` +/// +/// This script contains custom functions (`MyFunction()` and `MyOtherFunction()`) +/// as well as an `OnLibraryLoad()` function. `OnLibraryLoad()` is executed +/// whenever the library is loaded by `LoadLibrary()`; it calls +/// `RegisterLibraryScript()` to expose the names of the custom functions as +/// library scripts. When a library script is called with `RunLibraryScript()`, +/// the custom functions are called. +/// +/// If you want to do something more complicated that can't be handled by a +/// single function call, you can pass a unique number to +/// `RegisterLibraryScript()` as its second parameter, which will cause +/// `RunLibraryScript()` to call a special customizable dispatching function +/// called `OnLibraryScript()`. This function takes the name and number of the +/// desired function and executes the desired code. For example: +/// +/// ``` nwscript +/// #include "util_i_libraries" +/// +/// void OnLibraryLoad() +/// { +/// RegisterLibraryScript("Give50GP", 1); +/// RegisterLibraryScript("Give100GP", 2); +/// } +/// +/// void OnLibraryScript(string sScript, int nEntry) +/// { +/// switch (nEntry) +/// { +/// case 1: GiveGoldToCreature(OBJECT_SELF, 50); break; +/// case 2: GiveGoldToCreature(OBJECT_SELF, 100); break; +/// } +/// } +/// ``` +/// +/// **Note:** A library does not need to have a `main()` function, because this +/// will be automatically generated by the `LoadLibrary()` and +/// `RunLibraryScript()` functions. +/// +/// ## Using a Library +/// `util_i_libraries.nss` is needed to load or run library scripts. +/// +/// To use a library, you must first load it. This will activate the library's +/// `OnLibraryLoad()` function and register each desired function. +/// +/// ``` nwscript +/// // Loads a single library +/// LoadLibrary("my_l_library"); +/// +/// // Loads a CSV list of library scripts +/// LoadLibraries("pw_l_plugin, dlg_l_example, prr_l_main"); +/// +/// // Loads all libraries matching a glob pattern +/// LoadLibrariesByPattern("*_l_*"); +/// +/// // Loads all libraries matching a prefix +/// LoadLibrariesByPrefix("pw_l_"); +/// ``` +/// +/// If a library implements a script that has already been implemented in +/// another library, a warning will be issued and the newer script will take +/// precedence. +/// +/// Calling a library script is done using `RunLibraryScript()`. The name +/// supplied should be the name bound to the function in the library's +/// `OnLibraryLoad()`. If the name supplied is implemented by a library, the +/// library will be JIT compiled and the desired function will be called with +/// `ExecuteScriptChunk()`. Otherwise, the name will be assumed to match a +/// normal script, which will be executed with `ExecuteScript()`. +/// +/// ``` nwscript +/// // Executes a single library script on OBJECT_SELF +/// RunLibraryScript("MyFunction"); +/// +/// // Executes a CSV list of library scripts, for which oPC will be OBJECT_SELF +/// object oPC = GetFirstPC(); +/// RunLibraryScripts("MyFunction, MyOtherFunction", oPC); +/// ``` +/// +/// ## Pre-Compiled Libraries +/// By default, libraries are run using `ExecuteScriptChunk()`, which JIT +/// compiles the script and runs it each time the library script is called. If +/// you wish to have your script pre-compiled, you can include the script +/// `util_i_library.nss` in your file in place of `util_i_libraries.nss`. This +/// script contains a `main()` function that will call either your +/// `OnLibraryLoad()` or `OnLibraryScript()` function as appropriate; thus, if +/// you use this method, you *must* provide an `OnLibraryScript()` dispatch +/// function. +/// +/// **Note**: `util_i_library.nss` uses the nwnsc `default_function` pragma to +/// prevent compilation errors and will not compile with the toolset compiler. +/// If this is not desired, you can either comment those lines out or implement +/// the `main()` function yourself. +/// ---------------------------------------------------------------------------- + +#include "util_i_debug" +#include "util_i_csvlists" +#include "util_i_sqlite" +#include "util_i_nss" +#include "util_i_matching" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string LIB_RETURN = "LIB_RETURN"; ///< The return value of the library +const string LIB_LIBRARY = "LIB_LIBRARY"; ///< The library being processed +const string LIB_SCRIPT = "LIB_SCRIPT"; ///< The library script name +const string LIB_ENTRY = "LIB_ENTRY"; ///< The library script entry number + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Create a library table in the module's volatile sqlite database. +/// @param bReset if TRUE, the table will be dropped if already present. +/// @note This is called automatically by the library functions. +void CreateLibraryTable(int bReset = FALSE); + +/// @brief Add a database record associating a script with a library. +/// @param sLibrary The script to source from. +/// @param sScript The name to associate with the library script. +/// @param nEntry A number unique to sLibrary to identify this script. If this +/// is 0 and the library has not been pre-compiled, RunLibraryScript() will +/// call sScript directly. Otherwise, RunLibraryScript() will run a dispatch +/// function that can use this number to execute the correct code. Thus, +/// nEntry must be set if sScript does not exactly match the desired +/// function name or the function requires parameters. +void AddLibraryScript(string sLibrary, string sScript, int nEntry = 0); + +/// @brief Return the name of the library containing a script from the database. +/// @param sScript The name of the library script. +string GetScriptLibrary(string sScript); + +/// @brief Return the entry number associated with a library script. +/// @param sScript The name of the library script. +int GetScriptEntry(string sScript); + +/// @brief Return a prepared query with the with the library and entry data +/// associated with a library script. +/// @param sScript The name of the library script. +/// @note This allows users to retrive the same data returned by +/// GetScriptLibrary() and GetScriptEntry() with one function. +sqlquery GetScriptData(string sScript); + +/// @brief Return whether a script library has been loaded. +/// @param sLibrary The name of the script library file. +int GetIsLibraryLoaded(string sLibrary); + +/// @brief Load a script library by executing its OnLibraryLoad() function. +/// @param sLibrary The name of the script library file. +/// @param bForce If TRUE, will re-load the library if it was already loaded. +void LoadLibrary(string sLibrary, int bForce = FALSE); + +/// @brief Load a list of script libraries in sequence. +/// @param sLibraries A CSV list of libraries to load. +/// @param bForce If TRUE, will re-load the library if it was already loaded. +void LoadLibraries(string sLibraries, int bForce = FALSE); + +/// @brief Return a json array of script names with a prefix. +/// @param sPrefix The prefix matching the scripts to find. +/// @returns A sorted json array of script names, minus the extensions. +/// @note The search includes both nss and ncs files, with duplicates removed. +json GetScriptsByPrefix(string sPrefix); + +/// @brief Load all scripts matching the given glob pattern(s). +/// @param sPattern A CSV list of glob patterns to match with. Supported syntax: +/// - `*`: match zero or more characters +/// - `?`: match a single character +/// - `[abc]`: match any of a, b, or c +/// - `[a-z]`: match any character from a-z +/// - other text is matched literally +/// @param bForce If TRUE, will-reload the library if it was already loaded. +void LoadLibrariesByPattern(string sPattern, int bForce = FALSE); + +/// @brief Load all scripts with a given prefix as script libraries. +/// @param sPrefix A prefix for the desired script libraries. +/// @param bForce If TRUE, will re-load the library if it was already loaded. +/// @see GetMatchesPattern() for the rules on glob syntax. +void LoadLibrariesByPrefix(string sPrefix, int bForce = FALSE); + +/// @brief Execute a registered library script. +/// @param sScript The unique name of the library script. +/// @param oSelf The object that should execute the script as OBJECT_SELF. +/// @returns The integer value set with LibraryReturn() by sScript. +/// @note If sScript is not registered as a library script, it will be executed +/// as a regular script instead. +int RunLibraryScript(string sScript, object oSelf = OBJECT_SELF); + +/// @brief Execute a list of registered library scripts in sequence. +/// @param sScripts A CSV list of library script names. +/// @param oSelf The object that should execute the scripts as OBJECT_SELF. +/// @note If any script in sScripts is not registered as a library script, it +/// will be executed as a regular script instead. +void RunLibraryScripts(string sScripts, object oSelf = OBJECT_SELF); + +/// @brief Register a script to a library. The script can later be called using +/// RunLibraryScript(). +/// @param sScript A name for the script. Must be unique in the module. If a +/// second script with the same name is registered, it will overwrite the +/// first one. This value does not have to match the function or script name. +/// @param nEntry A number unique to this library to identify this script. If +/// this is 0 and the library has not been pre-compiled, RunLibraryScript() +/// will call sScript directly. Otherwise, RunLibraryScript() will run a +/// dispatch function that can use this number to execute the correct code. +/// Thus, nEntry must be set if sScript does not exactly match the desired +/// function name or the function requires parameters. +/// @note Must be called within a script library's OnLibraryLoad() function. For +/// uses in other places, use AddLibraryScript(). +void RegisterLibraryScript(string sScript, int nEntry = 0); + +/// @brief Set the return value of the currently executing library script. +/// @param nValue The value to return to the calling script. +void LibraryReturn(int nValue); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +void CreateLibraryTable(int bReset = FALSE) +{ + SqlCreateTableModule("library_scripts", + "id INTEGER PRIMARY KEY AUTOINCREMENT, " + + "sLibrary TEXT NOT NULL, " + + "sScript TEXT NOT NULL UNIQUE ON CONFLICT REPLACE, " + + "nEntry INTEGER NOT NULL);", + bReset); +} + +void AddLibraryScript(string sLibrary, string sScript, int nEntry = 0) +{ + CreateLibraryTable(); + + string sQuery = "INSERT INTO library_scripts (sLibrary, sScript, nEntry) " + + "VALUES (@sLibrary, @sScript, @nEntry);"; + sqlquery sql = SqlPrepareQueryModule(sQuery); + SqlBindString(sql, "@sLibrary", sLibrary); + SqlBindString(sql, "@sScript", sScript); + SqlBindInt(sql, "@nEntry", nEntry); + + SqlStep(sql); +} + +string GetScriptFieldData(string sField, string sScript) +{ + CreateLibraryTable(); + + string sQuery = "SELECT " + sField + " FROM library_scripts " + + "WHERE sScript = @sScript;"; + sqlquery sql = SqlPrepareQueryModule(sQuery); + SqlBindString(sql, "@sScript", sScript); + + return SqlStep(sql) ? SqlGetString(sql, 0) : ""; +} + +string GetScriptLibrary(string sScript) +{ + return GetScriptFieldData("sLibrary", sScript); +} + +int GetScriptEntry(string sScript) +{ + return StringToInt(GetScriptFieldData("nEntry", sScript)); +} + +sqlquery GetScriptData(string sScript) +{ + CreateLibraryTable(); + + string sQuery = "SELECT sLibrary, nEntry FROM library_scripts " + + "WHERE sScript = @sScript;"; + sqlquery sql = SqlPrepareQueryModule(sQuery); + SqlBindString(sql, "@sScript", sScript); + + return sql; +} + +int GetIsLibraryLoaded(string sLibrary) +{ + CreateLibraryTable(); + + string sQuery = "SELECT COUNT(sLibrary) FROM library_scripts " + + "WHERE sLibrary = @sLibrary LIMIT 1;"; + sqlquery sql = SqlPrepareQueryModule(sQuery); + SqlBindString(sql, "@sLibrary", sLibrary); + + return SqlStep(sql) ? SqlGetInt(sql, 0) : FALSE; +} + +void LoadLibrary(string sLibrary, int bForce = FALSE) +{ + Debug("Attempting to " + (bForce ? "force " : "") + "load library " + sLibrary); + + if (bForce || !GetIsLibraryLoaded(sLibrary)) + { + SetScriptParam(LIB_LIBRARY, sLibrary); + if (ResManGetAliasFor(sLibrary, RESTYPE_NCS) == "") + { + Debug(sLibrary + ".ncs not present; loading library as chunk"); + string sChunk = NssInclude(sLibrary) + NssVoidMain(NssFunction("OnLibraryLoad")); + string sError = ExecuteScriptChunk(sChunk, GetModule(), FALSE); + if (sError != "") + CriticalError("Could not load " + sLibrary + ": " + sError); + } + else + ExecuteScript(sLibrary, GetModule()); + + } + else + Error("Library " + sLibrary + " already loaded!"); +} + +void LoadLibraries(string sLibraries, int bForce = FALSE) +{ + Debug("Attempting to " + (bForce ? "force " : "") + "load libraries " + sLibraries); + + int i, nCount = CountList(sLibraries); + for (i = 0; i < nCount; i++) + LoadLibrary(GetListItem(sLibraries, i), bForce); +} + +// Private function for GetScriptsByPrefix*(). Adds all scripts of nResType +// matching a prefix to a json array and returns it. +json _GetScriptsByPrefix(json jArray, string sPrefix, int nResType) +{ + int i; + string sScript; + while ((sScript = ResManFindPrefix(sPrefix, nResType, ++i)) != "") + jArray = JsonArrayInsert(jArray, JsonString(sScript)); + + return jArray; +} + +json GetScriptsByPrefix(string sPrefix) +{ + json jScripts = _GetScriptsByPrefix(JsonArray(), sPrefix, RESTYPE_NCS); + jScripts = _GetScriptsByPrefix(jScripts, sPrefix, RESTYPE_NSS); + jScripts = JsonArrayTransform(jScripts, JSON_ARRAY_UNIQUE); + jScripts = JsonArrayTransform(jScripts, JSON_ARRAY_SORT_ASCENDING); + return jScripts; +} + +void LoadLibrariesByPattern(string sPatterns, int bForce = FALSE) +{ + if (sPatterns == "") + return; + + Debug("Finding libraries matching \"" + sPatterns + "\""); + json jPatterns = ListToJson(sPatterns); + json jLibraries = FilterByPatterns(GetScriptsByPrefix(""), jPatterns, TRUE); + LoadLibraries(JsonToList(jLibraries), bForce); +} + +void LoadLibrariesByPrefix(string sPrefix, int bForce = FALSE) +{ + Debug("Finding libraries with prefix \"" + sPrefix + "\""); + json jLibraries = GetScriptsByPrefix(sPrefix); + LoadLibraries(JsonToList(jLibraries), bForce); +} + +void LoadPrefixLibraries(string sPrefix, int bForce = FALSE) +{ + Debug("LoadPrefixLibraries() is deprecated; use LoadLibrariesByPrefix()"); + LoadLibrariesByPrefix(sPrefix, bForce); +} + +int RunLibraryScript(string sScript, object oSelf = OBJECT_SELF) +{ + if (sScript == "") return -1; + + string sLibrary; + int nEntry; + + sqlquery sqlScriptData = GetScriptData(sScript); + if (SqlStep(sqlScriptData)) + { + sLibrary = SqlGetString(sqlScriptData, 0); + nEntry = SqlGetInt(sqlScriptData, 1); + } + + DeleteLocalInt(oSelf, LIB_RETURN); + + if (sLibrary != "") + { + Debug("Library script " + sScript + " found in " + sLibrary + + (nEntry != 0 ? " at entry " + IntToString(nEntry) : "")); + + SetScriptParam(LIB_LIBRARY, sLibrary); + SetScriptParam(LIB_SCRIPT, sScript); + SetScriptParam(LIB_ENTRY, IntToString(nEntry)); + + if (ResManGetAliasFor(sLibrary, RESTYPE_NCS) == "") + { + Debug(sLibrary + ".ncs not present; running library script as chunk"); + string sChunk = NssInclude(sLibrary) + NssVoidMain(nEntry ? + NssFunction("OnLibraryScript", NssQuote(sScript) + ", " + IntToString(nEntry)) : + NssFunction(sScript)); + string sError = ExecuteScriptChunk(sChunk, oSelf, FALSE); + if (sError != "") + CriticalError("RunLibraryScript(" + sScript +") failed: " + sError); + } + else + ExecuteScript(sLibrary, oSelf); + } + else + { + Debug(sScript + " is not a library script; executing directly"); + ExecuteScript(sScript, oSelf); + } + + return GetLocalInt(oSelf, LIB_RETURN); +} + +void RunLibraryScripts(string sScripts, object oSelf = OBJECT_SELF) +{ + int i, nCount = CountList(sScripts); + for (i = 0; i < nCount; i++) + RunLibraryScript(GetListItem(sScripts, i), oSelf); +} + +void RegisterLibraryScript(string sScript, int nEntry = 0) +{ + string sLibrary = GetScriptParam(LIB_LIBRARY); + string sExist = GetScriptLibrary(sScript); + + if (sLibrary != sExist && sExist != "") + Warning(sLibrary + " is overriding " + sExist + "'s implementation of " + sScript); + + int nOldEntry = GetScriptEntry(sScript); + if (nOldEntry) + Warning(sLibrary + " already declared " + sScript + + " Old Entry: " + IntToString(nOldEntry) + + " New Entry: " + IntToString(nEntry)); + + AddLibraryScript(sLibrary, sScript, nEntry); +} + +void LibraryReturn(int nValue) +{ + SetLocalInt(OBJECT_SELF, LIB_RETURN, nValue); +} diff --git a/_module/nss/util_i_library.nss b/_module/nss/util_i_library.nss new file mode 100644 index 0000000..ff13385 --- /dev/null +++ b/_module/nss/util_i_library.nss @@ -0,0 +1,78 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_library.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Boilerplace code for creating a library dispatcher. Should only be +/// included in library scripts as it implements main(). +/// ---------------------------------------------------------------------------- + +#include "util_i_libraries" + +// ----------------------------------------------------------------------------- +// Function Protoypes +// ----------------------------------------------------------------------------- + +// This is a user-defined function that registers function names to a unique (to +// this library) number. When the function name is run using RunLibraryScript(), +// this number will be passed to the user-defined function OnLibraryScript(), +// which resolves the call to the correct function. +// +// Example usage: +// void OnLibraryLoad() +// { +// RegisterLibraryScript("MyFunction"); +// RegisterLibraryScript("MyOtherFunction"); +// } +// +// or, if using nEntry... +// void OnLibraryLoad() +// { +// RegisterLibraryScript("MyFunction", 1); +// RegisterLibraryScript("MyOtherFunction", 2); +// } +void OnLibraryLoad(); + +// This is a user-defined function that routes a unique (to the module) script +// name (sScript) or a unique (to this library) number (nEntry) to a function. +// +// Example usage: +// void OnLibraryScript(string sScript, int nEntry) +// { +// if (sScript == "MyFunction") MyFunction(); +// else if (sScript == "MyOtherFunction") MyOtherFunction(); +// } +// +// or, using nEntry... +// void OnLibraryScript(string sScript, int nEntry) +// { +// switch (nEntry) +// { +// case 1: MyFunction(); break; +// case 2: MyOtherFunction(); break; +// } +// } +// +// For advanced usage, see the libraries included in the Core Framework. +void OnLibraryScript(string sScript, int nEntry); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +// These are dummy implementations to prevent nwnsc from complaining that they +// do not exist. If you want to compile in the toolset rather than using nwnsc, +// comment these lines out. +// #pragma default_function(OnLibraryLoad) +// #pragma default_function(OnLibraryScript) + +// ----------------------------------------------------------------------------- +// Main Routine +// ----------------------------------------------------------------------------- + +void main() +{ + if (GetScriptParam(LIB_ENTRY) == "") + OnLibraryLoad(); + else + OnLibraryScript(GetScriptParam(LIB_SCRIPT), + StringToInt(GetScriptParam(LIB_ENTRY))); +} diff --git a/_module/nss/util_i_lists.nss b/_module/nss/util_i_lists.nss new file mode 100644 index 0000000..c34ab75 --- /dev/null +++ b/_module/nss/util_i_lists.nss @@ -0,0 +1,98 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_lists.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Compatibility functions to convert between CSV and localvar lists. +/// ---------------------------------------------------------------------------- + +#include "util_i_csvlists" +#include "util_i_varlists" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// Acceptable values for nListType in SplitList() and JoinList(). +const int LIST_TYPE_FLOAT = 0; +const int LIST_TYPE_INT = 1; +const int LIST_TYPE_STRING = 2; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Splits a comma-separated value list into a local variable list of the +/// given type. +/// @param oTarget Object on which to create the list +/// @param sList Source CSV list +/// @param sListName Name of the list to create or add to +/// @param bAddUnique If TRUE, prevents duplicate list items +/// @param nListType Type of list to create +/// LIST_TYPE_STRING (default) +/// LIST_TYPE_FLOAT +/// LIST_TYPE_INT +/// @returns JSON array of split CSV list +json SplitList(object oTarget, string sList, string sListName = "", int bAddUnique = FALSE, int nListType = LIST_TYPE_STRING); + +/// @brief Joins a local variable list of a given type into a comma-separated +/// value list +/// @param oTarget Object from which to source the local variable list +/// @param sListName Name of the local variable list +/// @param bAddUnique If TRUE, prevents duplicate list items +/// @param nListType Type of local variable list +/// LIST_TYPE_STRING (default) +/// LIST_TYPE_FLOAT +/// LIST_TYPE_INT +/// @returns Joined CSV list of local variable list +string JoinList(object oTarget, string sListName = "", int bAddUnique = FALSE, int nListType = LIST_TYPE_STRING); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +json SplitList(object oTarget, string sList, string sListName = "", int bAddUnique = FALSE, int nListType = LIST_TYPE_STRING) +{ + json jList = JSON_ARRAY; + + if (nListType == LIST_TYPE_STRING) + jList = ListToJson(sList, TRUE); + else + jList = JsonParse("[" + sList + "]"); + + string sListType = (nListType == LIST_TYPE_STRING ? VARLIST_TYPE_STRING : + nListType == LIST_TYPE_INT ? VARLIST_TYPE_INT : + VARLIST_TYPE_FLOAT); + + if (bAddUnique == TRUE) + jList = JsonArrayTransform(jList, JSON_ARRAY_UNIQUE); + + if (oTarget != OBJECT_INVALID) + _SetList(oTarget, sListType, sListName, jList); + + return jList; +} + +string JoinList(object oTarget, string sListName = "", int bAddUnique = FALSE, int nListType = LIST_TYPE_STRING) +{ + string sListType = (nListType == LIST_TYPE_STRING ? VARLIST_TYPE_STRING : + nListType == LIST_TYPE_INT ? VARLIST_TYPE_INT : + VARLIST_TYPE_FLOAT); + + json jList = _GetList(oTarget, sListType, sListName); + if (jList == JsonNull() || JsonGetLength(jList) == 0) + return ""; + + if (bAddUnique == TRUE) + jList = JsonArrayTransform(jList, JSON_ARRAY_UNIQUE); + + string sList; + if (nListType == LIST_TYPE_STRING) + sList = JsonToList(jList); + else + { + sList = JsonDump(jList); + sList = GetStringSlice(sList, 1, GetStringLength(sList) - 2); + } + + return sList; +} diff --git a/_module/nss/util_i_matching.nss b/_module/nss/util_i_matching.nss new file mode 100644 index 0000000..dbf9b75 --- /dev/null +++ b/_module/nss/util_i_matching.nss @@ -0,0 +1,128 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_matching.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Utilities for pattern matching. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Return whether a string matches a glob pattern. +/// @param sString The string to check. +/// @param sPattern A glob pattern. Supported syntax: +/// - `*`: match zero or more characters +/// - `?`: match a single character +/// - `[abc]`: match any of a, b, or c +/// - `[a-z]`: match any character from a-z +/// - other text is matched literally +/// @returns TRUE if sString matches sPattern; FALSE otherwise. +int GetMatchesPattern(string sString, string sPattern); + +/// @brief Return whether a string matches any of an array of glob patterns. +/// @param sString The string to check. +/// @param sPattern A json array of glob patterns. +/// @returns TRUE if sString matches sPattern; FALSE otherwise. +/// @see GetMatchesPattern() for supported glob syntax. +int GetMatchesPatterns(string sString, json jPatterns); + +/// @brief Return if any element of a json array matches a glob pattern. +/// @param jArray A json array of strings to check. +/// @param sPattern A glob pattern. +/// @param bNot If TRUE, will invert the selection, returning whether any +/// element does not match the glob pattern. +/// @returns TRUE if any element of jArray matches sPattern; FALSE otherwise. +/// @see GetMatchesPattern() for supported glob syntax. +int GetAnyMatchesPattern(json jArray, string sPattern, int bNot = FALSE); + +/// @brief Return if all elements of a json array match a glob pattern. +/// @param jArray A json array of strings to check. +/// @param sPattern A glob pattern. +/// @param bNot If TRUE, will invert the selection, returning whether all +/// elements do not match the glob pattern. +/// @returns TRUE if all elements of jArray match sPattern; FALSE otherwise. +/// @see GetMatchesPattern() for supported glob syntax. +int GetAllMatchesPattern(json jArray, string sPattern, int bNot = FALSE); + +/// @brief Filter out all elements of an array that do not match a glob pattern. +/// @param jArray A json array of strings to filter. +/// @param sPattern A glob pattern. +/// @param bNot If TRUE, will invert the selection, only keeping elements that +/// do not match the glob pattern. +/// @returns A modified copy of jArray with all non-matching elements removed. +/// @see GetMatchesPattern() for supported glob syntax. +json FilterByPattern(json jArray, string sPattern, int bNot = FALSE); + +/// @brief Filter out all elements of an array that do not match any of an array +/// of glob patterns. +/// @param jArray A json array of strings to filter. +/// @param jPatterns A json array of glob patterns. +/// @param bOrderByPatterns If TRUE, will order the results by the pattern they +/// matched with rather than by their placement in jArray. +/// @returns A modified copy of jArray with all non-matching elements removed. +/// @see GetMatchesPattern() for supported glob syntax. +json FilterByPatterns(json jArray, json jPatterns, int bOrderByPatterns = FALSE); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +int GetMatchesPattern(string sString, string sPattern) +{ + sqlquery q = SqlPrepareQueryObject(GetModule(), + "SELECT @string GLOB @pattern;"); + SqlBindString(q, "@string", sString); + SqlBindString(q, "@pattern", sPattern); + return SqlStep(q) ? SqlGetInt(q, 0) : FALSE; +} + +int GetMatchesPatterns(string sString, json jPatterns) +{ + sqlquery q = SqlPrepareQueryObject(GetModule(), + "SELECT 1 FROM json_each(@patterns) WHERE @value GLOB json_each.value;"); + SqlBindString(q, "@value", sString); + SqlBindJson(q, "@patterns", jPatterns); + return SqlStep(q) ? SqlGetInt(q, 0) : FALSE; +} + +int GetAnyMatchesPattern(json jArray, string sPattern, int bNot = FALSE) +{ + jArray = FilterByPattern(jArray, sPattern, bNot); + return JsonGetLength(jArray) != 0; +} + +int GetAllMatchesPattern(json jArray, string sPattern, int bNot = FALSE) +{ + return jArray == FilterByPattern(jArray, sPattern, bNot); +} + +json FilterByPattern(json jArray, string sPattern, int bNot = FALSE) +{ + if (!JsonGetLength(jArray)) + return jArray; + + sqlquery q = SqlPrepareQueryObject(GetModule(), + "SELECT json_group_array(value) FROM json_each(@array) " + + "WHERE value " + (bNot ? "NOT " : "") + "GLOB @pattern;"); + SqlBindJson(q, "@array", jArray); + SqlBindString(q, "@pattern", sPattern); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonArray(); +} + +json FilterByPatterns(json jArray, json jPatterns, int bOrderByPattern = FALSE) +{ + if (!JsonGetLength(jArray) || ! JsonGetLength(jPatterns)) + return jArray; + + sqlquery q = SqlPrepareQueryObject(GetModule(), + "SELECT json_group_array(value) FROM " + + "(SELECT DISTINCT v.key, v.value FROM " + + "json_each(@values) v JOIN " + + "json_each(@patterns) p " + + "WHERE v.value GLOB p.value " + + (bOrderByPattern ? "ORDER BY p.key);" : ");")); + SqlBindJson(q, "@values", jArray); + SqlBindJson(q, "@patterns", jPatterns); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonArray(); +} diff --git a/_module/nss/util_i_math.nss b/_module/nss/util_i_math.nss new file mode 100644 index 0000000..223299a --- /dev/null +++ b/_module/nss/util_i_math.nss @@ -0,0 +1,177 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_math.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Useful math utility functions. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Return the closest integer to the binary logarithm of a number. +int log2(int n); + +/// @brief Restrict an integer to a range. +/// @param nValue The number to evaluate. +/// @param nMin The minimum value for the number. +/// @param nMax The maximum value for the number. +/// @returns nValue if it is between nMin and nMax. Otherwise returns the +/// closest of nMin or nMax. +int clamp(int nValue, int nMin, int nMax); + +/// @brief Restrict a float to a range. +/// @param fValue The number to evaluate. +/// @param fMin The minimum value for the number. +/// @param fMax The maximum value for the number. +/// @returns fValue if it is between fMin and fMax. Otherwise returns the +/// closest of fMin or fMax. +float fclamp(float fValue, float fMin, float fMax); + +/// @brief Return the larger of two integers. +int max(int a, int b); + +/// @brief Return the smaller of two integers. +int min(int a, int b); + +/// @brief Return the sign of an integer. +/// @returns -1 if n is negative, 0 if 0, or 1 if positive. +int sign(int n); + +/// @brief Return the larger of two floats. +float fmax(float a, float b); + +/// @brief Return the smaller of two floats. +float fmin(float a, float b); + +/// @brief Return the sign of a float. +/// @returns -1 if f is negative, 0 if 0, or 1 if positive. +int fsign(float f); + +/// @brief Truncate a float (i.e., remove numbers to the right of the decimal +/// point). +float trunc(float f); + +/// @brief Return the fractional part of a float (i.e., numbers to the right of +/// the decimal point). +float frac(float f); + +/// @brief Return a % b (modulo function). +/// @param a The dividend +/// @param b The divisor +/// @note For consistency with NWN's integer modulo operator, the result has the +/// same sign as a (i.e., fmod(-1, 2) == -1). +float fmod(float a, float b); + +/// @brief Round a float down to the nearest whole number. +float floor(float f); + +/// @brief Round a float up to the nearest whole number. +float ceil(float f); + +/// @brief Round a float towards to the nearest whole number. +/// @note In case of a tie (i.e., +/- 0.5), rounds away from 0. +float round(float f); + +/// @brief Determine if x is in [a..b] +/// @param x Value to compare +/// @param a Low end of range +/// @param b High end of range +int between(int x, int a, int b); + +/// @brief Determine if x is in [a..b] +/// @param x Value to compare +/// @param a Low end of range +/// @param b High end of range +int fbetween(float x, float a, float b); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +int log2(int n) +{ + int nResult; + while (n >>= 1) + nResult++; + return nResult; +} + +int clamp(int nValue, int nMin, int nMax) +{ + return (nValue < nMin) ? nMin : ((nValue > nMax) ? nMax : nValue); +} + +float fclamp(float fValue, float fMin, float fMax) +{ + return (fValue < fMin) ? fMin : ((fValue > fMax) ? fMax : fValue); +} + +int max(int a, int b) +{ + return (b > a) ? b : a; +} + +int min(int a, int b) +{ + return (b > a) ? a : b; +} + +int sign(int n) +{ + return (n > 0) ? 1 : (n < 0) ? -1 : 0; +} + +float fmax(float a, float b) +{ + return (b > a) ? b : a; +} + +float fmin(float a, float b) +{ + return (b > a) ? a : b; +} + +int fsign(float f) +{ + return f > 0.0 ? 1 : f < 0.0 ? -1 : 0; +} + +float trunc(float f) +{ + return IntToFloat(FloatToInt(f)); +} + +float frac(float f) +{ + return f - trunc(f); +} + +float fmod(float a, float b) +{ + return a - b * trunc(a / b); +} + +float floor(float f) +{ + return IntToFloat(FloatToInt(f) - (f < 0.0)); +} + +float ceil(float f) +{ + return IntToFloat(FloatToInt(f) + (trunc(f) < f)); +} + +float round(float f) +{ + return IntToFloat(FloatToInt(f + (f < 0.0 ? -0.5 : 0.5))); +} + +int between(int x, int a, int b) +{ + return ((x - a) * (x - b)) <= 0; +} + +int fbetween(float x, float a, float b) +{ + return ((x - a) * (x - b)) <= 0.0; +} diff --git a/_module/nss/util_i_nss.nss b/_module/nss/util_i_nss.nss new file mode 100644 index 0000000..8640db9 --- /dev/null +++ b/_module/nss/util_i_nss.nss @@ -0,0 +1,224 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_nss.nss +/// @author Daz +/// @brief Functions to assemble scripts for use with `ExecuteScriptChunk()`. +/// @note Borrowed from https://github.com/Daztek/EventSystem +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Return a `void main()` block. +/// @param sContents The contents of the block. +string NssVoidMain(string sContents); + +/// @brief Return an `int StartingConditional()` block. +/// @param sContents The contents of the block. +string NssStartingConditional(string sContents); + +/// @brief Return an include directive. +/// @param sIncludeFile The file to include. +string NssInclude(string sIncludeFile); + +/// @brief Return an if statement with a comparison. +/// @param sLeft The left side of the comparison. If sComparison or sRight are +/// blank, will be evalated as a boolean expression. +/// @param sComparison The comparison operator. +/// @param sRight The right side of the comparison. +string NssIf(string sLeft, string sComparison = "", string sRight = ""); + +/// @brief Return an else statement. +string NssElse(); + +/// @brief Return an else-if statement with a comparison. +/// @param sLeft The left side of the comparison. If sComparison or sRight are +/// blank, will be evalated as a boolean expression. +/// @param sComparison The comparison operator. +/// @param sRight The right side of the comparison. +string NssElseIf(string sLeft, string sComparison = "", string sRight = ""); + +/// @brief Create a while statement with a comparison. +/// @param sLeft The left side of the comparison. If sComparison or sRight are +/// blank, will be evalated as a boolean expression. +/// @param sComparison The comparison operator. +/// @param sRight The right side of the comparison. +string NssWhile(string sLeft, string sComparison = "", string sRight = ""); + +/// @brief Return a script block bounded by curly brackets. +/// @param sContents The contents of the block. +string NssBrackets(string sContents); + +/// @brief Return a string wrapped in double quotes. +/// @param sString The string to wrap. +string NssQuote(string sString); + +/// @brief Return a switch statement. +/// @param sVariable The variable to evaluate in the switch statement. +/// @param sCases A series of case statements the switch should dispatch to. +/// @see NssCase(). +string NssSwitch(string sVariable, string sCases); + +/// @brief Return a case statement. +/// @param nCase The value matching the switch statement. +/// @param sContents The contents of the case block. +/// @param bBreak If TRUE, will add a break statement after sContents. +string NssCase(int nCase, string sContents, int bBreak = TRUE); + +/// @brief Return an object variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssObject(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return a string variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssString(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return an int variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssInt(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return a float variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssFloat(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return a vector variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssVector(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return a location variable declaration and/or assignment. +/// @param sVarName The name for the variable. +/// @param sValue The value to assign to the variable. If blank, no value will +/// be assigned. +/// @param bIncludeType If TRUE, the variable will be declared as well. +string NssLocation(string sVarName, string sValue = "", int bIncludeType = TRUE); + +/// @brief Return a call, prototype, or definition of a function. +/// @param sFunction The name of the function. +/// @param sArguments The list of arguments for the function. +/// @param bAddSemicolon If TRUE, a semicolon will be assed to the end of the +/// statement. +string NssFunction(string sFunction, string sArguments = "", int bAddSemicolon = TRUE); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +string NssVoidMain(string sContents) +{ + return "void main() { " + sContents + " }"; +} + +string NssStartingConditional(string sContents) +{ + return "int StartingConditional() { return " + sContents + " }"; +} + +string NssInclude(string sIncludeFile) +{ + return sIncludeFile == "" ? sIncludeFile : "#" + "include \"" + sIncludeFile + "\" "; +} + +string NssCompare(string sLeft, string sComparison, string sRight) +{ + return (sComparison == "" || sRight == "") ? sLeft : sLeft + " " + sComparison + " " + sRight; +} + +string NssIf(string sLeft, string sComparison = "", string sRight = "") +{ + return "if (" + NssCompare(sLeft, sComparison, sRight) + ") "; +} + +string NssElse() +{ + return "else "; +} + +string NssElseIf(string sLeft, string sComparison = "", string sRight = "") +{ + return "else if (" + NssCompare(sLeft, sComparison, sRight) + ") "; +} + +string NssWhile(string sLeft, string sComparison = "", string sRight = "") +{ + return "while (" + NssCompare(sLeft, sComparison, sRight) + ") "; +} + +string NssBrackets(string sContents) +{ + return "{ " + sContents + " } "; +} + +string NssQuote(string sString) +{ + return "\"" + sString + "\""; +} + +string NssSwitch(string sVariable, string sCases) +{ + return "switch (" + sVariable + ") { " + sCases + " }"; +} + +string NssCase(int nCase, string sContents, int bBreak = TRUE) +{ + return "case " + IntToString(nCase) + ": { " + sContents + (bBreak ? " break;" : "") + " } "; +} + +string NssSemicolon(string sString) +{ + return (GetStringRight(sString, 1) == ";" || GetStringRight(sString, 2) == "; ") ? sString + " " : sString + "; "; +} + +string NssVariable(string sType, string sVarName, string sValue) +{ + return sType + " " + sVarName + (sValue == "" ? "; " : " = " + NssSemicolon(sValue)); +} + +string NssObject(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "object" : "", sVarName, sValue); +} + +string NssString(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "string" : "", sVarName, sValue); +} + +string NssInt(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "int" : "", sVarName, sValue); +} + +string NssFloat(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "float" : "", sVarName, sValue); +} + +string NssVector(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "vector" : "", sVarName, sValue); +} + +string NssLocation(string sVarName, string sValue = "", int bIncludeType = TRUE) +{ + return NssVariable(bIncludeType ? "location" : "", sVarName, sValue); +} + +string NssFunction(string sFunction, string sArguments = "", int bAddSemicolon = TRUE) +{ + return sFunction + "(" + sArguments + (bAddSemicolon ? ");" : ")") + " "; +} diff --git a/_module/nss/util_i_sqlite.nss b/_module/nss/util_i_sqlite.nss new file mode 100644 index 0000000..ac62990 --- /dev/null +++ b/_module/nss/util_i_sqlite.nss @@ -0,0 +1,175 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_sqlite.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Helper functions for NWN:EE SQLite databases. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Alias for `SqlPreparerQueryObject(GetModule(), sSQL)`. +sqlquery SqlPrepareQueryModule(string sSQL); + +/// @brief Prepares and executes a query on a PC's peristent database. +/// @param oPC The PC that stores the database. +/// @param sQuery The SQL statement to execute. +/// @returns Whether the query was successful. +int SqlExecPC(object oPC, string sQuery); + +/// @brief Prepares and executes a query on the module's volatile database. +/// @param sQuery The SQL statement to execute. +/// @returns Whether the query was successful. +int SqlExecModule(string sQuery); + +/// @brief Prepares and executes a query on a persistent campaign database. +/// @param sDatabase The name of the campaign database file (minus extension). +/// @param sQuery The SQL statement to execute. +/// @returns Whether the query was successful. +int SqlExecCampaign(string sDatabase, string sQuery); + +/// @brief Creates a table in a PC's persistent database. +/// @param oPC The PC that stores the database. +/// @param sTable The name of the table. +/// @param sStructure The SQL describing the structure of the table (i.e., +/// everything that would go between the parentheses). +/// @param bForce Whether to drop an existing table. +void SqlCreateTablePC(object oPC, string sTable, string sStructure, int bForce = FALSE); + +/// @brief Creates a table in the module's volatile database. +/// @param sTable The name of the table. +/// @param sStructure The SQL describing the structure of the table (i.e., +/// everything that would go between the parentheses). +/// @param bForce Whether to drop an existing table. +void SqlCreateTableModule(string sTable, string sStructure, int bForce = FALSE); + +/// @brief Creates a table in a persistent campaign database. +/// @param sDatabase The name of the campaign database file (minus extension). +/// @param sTable The name of the table. +/// @param sStructure The SQL describing the structure of the table (i.e., +/// everything that would go between the parentheses). +/// @param bForce Whether to drop an existing table. +void SqlCreateTableCampaign(string sDatabase, string sTable, string sStructure, int bForce = FALSE); + +/// @brief Checks if a table exists the PC's persistent database. +/// @param oPC The PC that stores the database. +/// @param sTable The name of the table to check for. +/// @returns Whether the table exists. +int SqlGetTableExistsPC(object oPC, string sTable); + +/// @brief Checks if a table exists in the module's volatile database. +/// @param sTable The name of the table to check for. +/// @returns Whether the table exists. +int SqlGetTableExistsModule(string sTable); + +/// @brief Checks if a table exists in a peristent campaign database. +/// @param sDatabase The name of the campaign database file (minus extension). +/// @param sTable The name of the table to check for. +/// @returns Whether the table exists. +int SqlGetTableExistsCampaign(string sDatabase, string sTable); + +/// @brief Gets the ID of the last row inserted into a PC's persistent database. +/// @param oPC The PC that stores the database. +/// @returns The ID of the last inserted row or -1 on error. +int SqlGetLastInsertIdPC(object oPC); + +/// @brief Gets the ID of the last row inserted into the module's volatile +/// database. +/// @returns The ID of the last inserted row or -1 on error. +int SqlGetLastInsertIdModule(); + +/// @brief Gets the ID of the last row inserted into a persistent campaign +/// database. +/// @param sDatabase The name of the campaign database file (minus extension). +/// @returns The ID of the last inserted row or -1 on error. +int SqlGetLastInsertIdCampaign(string sDatabase); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +sqlquery SqlPrepareQueryModule(string sSQL) +{ + return SqlPrepareQueryObject(GetModule(), sSQL); +} + +int SqlExecPC(object oPC, string sQuery) +{ + return SqlStep(SqlPrepareQueryObject(oPC, sQuery)); +} + +int SqlExecModule(string sQuery) +{ + return SqlStep(SqlPrepareQueryModule(sQuery)); +} + +int SqlExecCampaign(string sDatabase, string sQuery) +{ + return SqlStep(SqlPrepareQueryCampaign(sDatabase, sQuery)); +} + +void SqlCreateTablePC(object oPC, string sTable, string sStructure, int bForce = FALSE) +{ + if (bForce) + SqlExecPC(oPC, "DROP TABLE IF EXISTS " + sTable + ";"); + + SqlExecPC(oPC, "CREATE TABLE IF NOT EXISTS " + sTable + "(" + sStructure + ");"); +} + +void SqlCreateTableModule(string sTable, string sStructure, int bForce = FALSE) +{ + if (bForce) + SqlExecModule("DROP TABLE IF EXISTS " + sTable + ";"); + + SqlExecModule("CREATE TABLE IF NOT EXISTS " + sTable + "(" + sStructure + ");"); +} + +void SqlCreateTableCampaign(string sDatabase, string sTable, string sStructure, int bForce = FALSE) +{ + if (bForce) + SqlExecCampaign(sDatabase, "DROP TABLE IF EXISTS " + sTable + ";"); + + SqlExecCampaign(sDatabase, "CREATE TABLE IF NOT EXISTS " + sTable + "(" + sStructure + ");"); +} + +int SqlGetTableExistsPC(object oPC, string sTable) +{ + string sQuery = "SELECT name FROM sqlite_master WHERE type='table' AND name = @table;"; + sqlquery qQuery = SqlPrepareQueryObject(oPC, sQuery); + SqlBindString(qQuery, "@table", sTable); + return SqlStep(qQuery); +} + +int SqlGetTableExistsModule(string sTable) +{ + string sQuery = "SELECT name FROM sqlite_master WHERE type='table' AND name = @table;"; + sqlquery qQuery = SqlPrepareQueryModule(sQuery); + SqlBindString(qQuery, "@table", sTable); + return SqlStep(qQuery); +} + +int SqlGetTableExistsCampaign(string sDatabase, string sTable) +{ + string sQuery = "SELECT name FROM sqlite_master WHERE type='table' AND name = @table;"; + sqlquery qQuery = SqlPrepareQueryCampaign(sDatabase, sQuery); + SqlBindString(qQuery, "@table", sTable); + return SqlStep(qQuery); +} + +int SqlGetLastInsertIdPC(object oPC) +{ + sqlquery qQuery = SqlPrepareQueryObject(oPC, "SELECT last_insert_rowid();"); + return SqlStep(qQuery) ? SqlGetInt(qQuery, 0) : -1; +} + +int SqlGetLastInsertIdModule() +{ + sqlquery qQuery = SqlPrepareQueryModule("SELECT last_insert_rowid();"); + return SqlStep(qQuery) ? SqlGetInt(qQuery, 0) : -1; +} + +int SqlGetLastInsertIdCampaign(string sDatabase) +{ + sqlquery qQuery = SqlPrepareQueryCampaign(sDatabase, "SELECT last_insert_rowid();"); + return SqlStep(qQuery) ? SqlGetInt(qQuery, 0) : -1; +} diff --git a/_module/nss/util_i_strftime.nss b/_module/nss/util_i_strftime.nss new file mode 100644 index 0000000..b60620a --- /dev/null +++ b/_module/nss/util_i_strftime.nss @@ -0,0 +1,1060 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_strftime.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Functions for formatting times. +/// ---------------------------------------------------------------------------- +/// @details This file contains an implementation of C's strftime() in nwscript. +/// +/// # Formatting +/// +/// You can format a Time using the `strftime()` function. This function takes +/// a Time as the first parameter (`t`) and a *format specification string* +/// (`sFormat`) as the second parameter. The format specification string may +/// contain special character sequences called *conversion specifications*, each +/// of which is introduced by the `%` character and terminated by some other +/// character known as a *conversion specifier character*. All other character +/// sequences are *ordinary character sequences*. +/// +/// The characters of ordinary character sequences are copied verbatim from +/// `sFormat` to the returned value. However, the characters of conversion +/// specifications are replaced as shown in the list below. Some sequences may +/// have their output customized using a *locale*, which can be passed using the +/// third parameter of `strftime()` (`sLocale`). +/// +/// Several aliases for `strftime()` exist. `FormatTime()`, `FormatDate()`, and +/// `FormatDateTime()` each take a calendar Time and will default to formatting +/// to a locale-specific representation of the time, date, or date and time +/// respectively. `FormatDuration()` takes a duration Time and defaults to +/// showing an ISO 8601 formatted datetime with a sign character before it. +/// +/// ## Conversion Specifiers +/// - `%a`: The abbreviated name of the weekday according to the current locale. +/// The specific names used in the current locale can be set using the +/// key `LOCALE_DAYS_ABBR`. If no abbreviated names are available in the +/// locale, will fall back to the full day name. +/// - `%A`: The full name of the weekday according to the current locale. +/// The specific names used in the current locale can be set using the +/// key `LOCALE_DAYS`. +/// - `%b`: The abbreviated name of the month according to the current locale. +/// The specific names used in the current locale can be set using the +/// key `LOCALE_MONTHS_ABBR`. If no abbreviated names are available in +/// the locale, will fall back to the full month name. +/// - `%B`: The full name of the month according to the current locale. The +/// specific names used in the current locale can be set using the key +/// `LOCALE_MONTHS`. +/// - `%c`: The preferred date and time representation for the current locale. +/// The specific format used in the current locale can be set using the +/// key `LOCALE_DATETIME_FORMAT` for the `%c` conversion specification +/// and `ERA_DATETIME_FORMAT` for the `%Ec` conversion specification. +/// With the default settings, this is equivalent to `%Y-%m-%d +/// %H:%M:%S:%f`. This is the default value of `sFormat` for +/// `FormatDateTime()`. +/// - `%C`: The century number (year / 100) as a 2-or-3-digit integer (00..320). +/// (The `%EC` conversion specification corresponds to the name of the +/// era, which can be set using the era key `ERA_NAME`.) +/// - `%d`: The day of the month as a 2-digit decimal number (01..28). +/// - `%D`: Equivalent to `%m/%d/%y`, the standard US time format. Note that +/// this may be ambiguous and confusing for non-Americans. +/// - `%e`: The day of the month as a decimal number, but a leading zero is +/// replaced by a space. Equivalent to `%_d`. +/// - `%E`: Modifier: use alternative "era-based" format (see below). +/// - `%f`: The millisecond as a 3-digit decimal number (000..999). +/// - `%F`: Equivalent to `%Y-%m-%d`, the ISO 8601 date format. +/// - `%H`: The hour (24-hour clock) as a 2-digit decimal number (00..23). +/// - `%I`: The hour (12-hour clock) as a 2-digit decimal number (01..12). +/// - `%j`: The day of the year as a 3-digit decimal number (000..336). +/// - `%k`: The hour (24-hour clock) as a decimal number (0..23). Single digits +/// are preceded by a space. Equivalent to `%_H`. +/// - `%l`: The hour (12-hour clock) as a decimal number (1..12). Single digits +/// are preceded by a space. Equivalent to `%_I`. +/// - `%m`: The month as a 2-digit decimal number (01..12). +/// - `%M`: The minute as a 2-digit decimal number (00..59, depending on +/// `t.MinsPerHour`). +/// - `%O`: Modifier: use ordinal numbers (1st, 2nd, etc.) (see below). +/// - `%p`: Either "AM" or "PM" according to the given Time, or the +/// corresponding values from the locale. The specific word used can be +/// set for the current locale using the key `LOCALE_AMPM`. +/// - `%P`: Like `%p`, but lowercase. Yes, it's silly that it's not the other +/// way around. +/// - `%r`: The preferred AM/PM time representation for the current locale. The +/// specific format used in the current locale can be set using the key +/// `LOCALE_AMPM_FORMAT`. With the default settings, this is equivalent +/// to `%I:%M:%S %p`. +/// - `%R`: The time in 24-hour notation. Equivalent to `%H:%M`. For a version +/// including seconds, see `%T`. +/// - `%S`: The second as a 2-digit decimal number (00..59). +/// - `%T`: The time in 24-hour notation. Equivalent to `%H:%M:%S`. For a +/// version without seconds, see `%R`. +/// - `%u`: The day of the week as a 1-indexed decimal (1..7). +/// - `%w`: The day of the week as a 0-indexed decimal (0..6). +/// - `%x`: The preferred date representation for the current locale without the +/// time. The specific format used in the current locale can be set +/// using the key `LOCALE_TIME_FORMAT` for the `%x` conversion +/// specification and `ERA_TIME_FORMAT` for the `%Ex` conversion +/// specification. With the default settings, this is equivalent to +/// `%Y-%m-%d`. This is the default value of `sFormat` for +/// `FomatDate()`. +/// - `%X`: The preferred time representation for the current locale without the +/// date. The specific format used in the current locale can be set +/// using the key `LOCALE_DATE_FORMAT` for the `%X` conversion +/// specification and `ERA_DATE_FORMAT` for the `%EX` conversion +/// specification. With the default settings, this is equivalent to +/// `%H:%M:%S`. This is the default value of `sFormat` for +/// `FormatTime()`. +/// - `%y`: The year as a 2-digit decimal number without the century (00..99). +/// (The `%Ey` conversion specification corresponds to the year since +/// the beginning of the era denoted by the `%EC` conversion +/// specification.) +/// - `%Y`: The year as a decimal number including the century (0000..32000). +/// (The `%EY` conversion specification corresponds to era key +/// `ERA_FORMAT`; with the default era settings, this is equivalent to +/// `%Ey %EC`.) +/// - `%%`: A literal `%` character. +/// +/// ## Modifier Characters +/// Some conversion specifications can be modified by preceding the conversion +/// specifier character by the `E` or `O` *modifier* to indicate that an +/// alternative format should be used. If the alternative format does not exist +/// for the locale, the behavior will be as if the unmodified conversion +/// specification were used. +/// +/// The `E` modifier signifies using an alternative era-based representation. +/// The following are valid: `%Ec`, `%EC`, `%Ex`, `%EX`, `%Ey`, and `%EY`. +/// +/// The `O` modifier signifies representing numbers in ordinal form (e.g., 1st, +/// 2nd, etc.). The ordinal suffixes for each number can be set using the locale +/// key `LOCALE_ORDINAL_SUFFIXES`. The following are valid: `%Od`, `%Oe`, `%OH`, +/// `%OI`, `%Om`, `%OM`, `%OS`, `%Ou`, `%Ow`, `%Oy`, and `%OY`. +/// +/// ## Flag Characters +/// Between the `%` character and the conversion specifier character, an +/// optional *flag* and *field width* may be specified. (These should precede +/// the `E` or `O` characters, if present). +/// +/// The following flag characters are permitted: +/// - `_`: (underscore) Pad a numeric result string with spaces. +/// - `-`: (dash) Do not pad a numeric result string. +/// - `0`: Pad a numeric result string with zeroes even if the conversion +/// specifier character uses space-padding by default. +/// - `^`: Convert alphabetic characters in the result string to uppercase. +/// - `+`: Display a `-` before numeric values if the Time is negative, or a `+` +/// if the Time is positive or 0. +/// - `,`: Add comma separators for long numeric values. +/// +/// An optional decimal width specifier may follow the (possibly absent) flag. +/// If the natural size of the field is smaller than this width, the result +/// string is padded (on the left) to the specified width. The string is never +/// truncated. +/// +/// ## Examples +/// +/// ```nwscript +/// struct Time t = StringToTime("1372-06-01 13:00:00:000"); +/// +/// // Default formatting +/// FormatDateTime(t); // "1372-06-01 13:00:00:000" +/// FormatDate(t); // "1372-06-01" +/// FormatTime(t); // "13:00:00:000" +/// +/// // Using custom formats +/// FormatTime(t, "Today is %A, %B %Od."); // "Today is Monday, June 1st." +/// FormatTime(t, "%I:%M %p"); // "01:00 PM" +/// FormatTime(t, "%-I:%M %p"); // "1:00 PM" +/// ``` +/// ---------------------------------------------------------------------------- +/// # Advanced Usage +/// +/// ## Locales +/// +/// A locale is a json object that contains localization settings for formatting +/// functions. A default locale will be constructed using the configuration +/// values in `util_c_times.nss`, but you can also construct locales yourself. +/// An application for this might be having different areas in the module use +/// different month or day names, etc. +/// +/// A locale is a simple json object: +/// ```nwscript +/// json jLocale = JsonObject(); +/// ``` +/// +/// Alternatively, you can initialize a locale with the default values from +/// util_c_times: +/// ```nwscript +/// json jLocale = NewLocale(); +/// ``` +/// +/// Keys are then added using `SetLocaleString()`: +/// ```nwscript +/// jLocale = SetLocaleString(jLocale, LOCALE_DAYS, "Moonday, Treeday, etc."); +/// ``` +/// +/// Keys can be retrieved using `GetLocaleString()`, which takes an optional +/// default value if the key is not set: +/// ```nwscript +/// string sDays = GetLocaleString(jLocale, LOCALE_DAYS); +/// string sDaysAbbr = GetLocaleString(jLocale, LOCALE_DAYS_ABBR, sDays); +/// ``` +/// +/// Locales can be saved with a name. That names can then be passed to +/// formatting functions: +/// ```nwscript +/// json jLocale = JsonObject(); +/// jLocale = SetLocaleString(jLocale, LOCALE_DAYS, "Moonday, Treeday, Heavensday, Valarday, Shipday, Starday, Sunday"); +/// jLocale = SetLocaleString(jLocale, LOCALE_MONTHS, "Narvinye, Nenime, Sulime, Varesse, Lotesse, Narie, Cermie, Urime, Yavannie, Narquelie, Hisime, Ringare"); +/// SetLocale(jLocale, "ME"); +/// FormatTime(t, "Today is %A, %B %Od."); // "Today is Monday, June 1st +/// FormatTime(t, "Today is %A, %B %Od.", "ME"); // "Today is Moonday, Narie 1st +/// ``` +/// +/// You can change the default locale so that you don't have to pass the name +/// every time: +/// ```nwscript +/// SetDefaultLocale("ME"); +/// FormatTime(t, "Today is %A, %B %Od."); // "Today is Moonday, Narie 1st +/// ``` +/// +/// The following keys are currently supported: +/// - `LOCALE_DAYS`: a CSV list of 7 weekday names. Accessed by `%A`. +/// - `LOCALE_DAYS_ABBR`: a CSV list of 7 abbreviated weekday names. If not set, +/// the `FormatTime()` function will use `LOCALE_DAYS` instead. Accessed by +/// `%a`. +/// - `LOCALE_MONTHS`: a CSV list of 12 month names. Accessed by `%B`. +/// - `LOCALE_MONTHS_ABBR`: a CSV list of 12 abbreviated month names. If not +/// set, the `FormatTime()` function will use `LOCALE_MONTHS` instead. +/// Accessed by `%b`. +/// - `LOCALE_AMPM`: a CSV list of 2 AM/PM elements. Accessed by `%p` and `%P`. +/// - `LOCALE_ORDINAL_SUFFIXES`: a CSV list of suffixes for constructing ordinal +/// numbers. See util_c_times's documentation of `DEFAULT_ORDINAL_SUFFIXES` +/// for details. +/// - `LOCALE_DATETIME_FORMAT`: a date and time format string. Aliased by `%c`. +/// - `LOCALE_DATE_FORMAT`: a date format string. Aliased by `%x`. +/// - `LOCALE_TIME_FORMAT`: a time format string. Aliased by `%X`. +/// - `LOCALE_AMPM_FORMAT`: a time format string using AM/PM form. Aliased +/// by `%r`. +/// - `ERA_DATETIME_FORMAT`: a format string to display the date and time. If +/// not set, will fall back to `LOCALE_DATETIME_FORMAT`. Aliased by `%Ec`. +/// - `ERA_DATE_FORMAT`: a format string to display the date without the time. +/// If not set, will fall back to `LOCALE_DATE_FORMAT`. Aliased by `%Ex`. +/// - `ERA_TIME_FORMAT`: a format string to display the time without the date. +/// If not set, will fall back to `LOCALE_TIME_FORMAT`. Aliased by `%EX`. +/// - `ERA_YEAR_FORMAT`: a format string to display the year. If not set, will +/// display the year. Aliased by `%EY`. +/// - `ERA_NAME`: the name of an era. If not set and no era matches the current +/// year, will display the century. Aliased by `%EC`. +/// +/// ## Eras +/// Locales can also hold an array of eras. Eras are json objects which name a +/// time range. When formatting using the `%E` modifier, the start Times of each +/// era in the array are compared to the Time to be formatted; the era with the +/// latest start that is still before the Time is selected. Format codes can +/// then refer to the era's name, year relative to the era start, and other +/// era-specific formats. +/// +/// An era can be created using `DefineEra()`. This function takes a name and a +/// start Time. See the documentation for `DefineEra()` for further info: +/// ```nwscript +/// // Create an era that begins at the first possible calendar time +/// json jFirst = DefineEra("First Age", GetTime()); +/// +/// // Create an era that begins on a particular year +/// json jSecond = DefineEra("Second Age", GetTime(590)); +/// ``` +/// +/// The `{Get/Set}LocaleString()` functions also apply to eras: +/// ```nwscript +/// jSecond = SetLocaleString(jSecond, ERA_DATETIME_FORMAT, "%B %Od, %EY"); +/// jSecond = SetLocaleString(jSecond, ERA_YEAR_FORMAT, "%EY 2E"); +/// ``` +/// +/// You can add an era to a locale using `AddEra()`: +/// ```nwscript +/// json jLocale = GetLocale("ME"); +/// jLocale = SetLocaleString(jLocale, LOCALE_DAYS, "Moonday, Treeday, Heavensday, Valarday, Shipday, Starday, Sunday"); +/// jLocale = SetLocaleString(jLocale, LOCALE_MONTHS, "Narvinye, Nenime, Sulime, Varesse, Lotesse, Narie, Cermie, Urime, Yavannie, Narquelie, Hisime, Ringare"); +/// jLocale = AddEra(jLocale, jFirst); +/// jLocale = AddEra(jLocale, jSecond); +/// SetLocale(jLocale, "ME"); +/// ``` +/// +/// You can then access the era settings using the `%E` modifier: +/// ```nwscript +/// FormatTime(t, "Today is %A, %B %Od, %EY.", "ME"); // "Today is Moonday, Narie 1st, 783 2E." +/// +/// // You can combine the `%E` and `%O` modifiers +/// FormatTime(t, "It is the %EOy year of the %EC.", "ME"); // "It is the 783rd year of the Second Age." +/// ``` +/// +/// The following keys are available to eras: +/// - `ERA_NAME`: the name of the era. Aliased by `%EC`. +/// - `ERA_DATETIME_FORMAT`: a format string to display the date and time. If +/// not set, will fall back to the value on the locale. Aliased by `%Ec`. +/// - `ERA_DATE_FORMAT`: a format string to display the date without the time. +/// If not set, will fall back to the value on the locale. Aliased by `%Ex`. +/// - `ERA_TIME_FORMAT`: a format string to display the time without the date. +/// If not set, will fall back to the value on the locale. Aliased by `%EX`. +/// - `ERA_YEAR_FORMAT`: a format string to display the year. Defaults to +/// `%Ey %EC`. If not set, will fall back to the value on the locale. Aliased +/// by `%EY`. +/// ---------------------------------------------------------------------------- + +#include "util_i_times" +#include "util_i_csvlists" +#include "util_c_strftime" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// These are the characters used as flags in time format codes. +const string TIME_FLAG_CHARS = "EO^,+-_0123456789"; + +const int TIME_FLAG_ERA = 0x01; ///< `E`: use era-based formatting +const int TIME_FLAG_ORDINAL = 0x02; ///< `O`: use ordinal numbers +const int TIME_FLAG_UPPERCASE = 0x04; ///< `^`: use uppercase letters +const int TIME_FLAG_COMMAS = 0x08; ///< `,`: add comma separators +const int TIME_FLAG_SIGN = 0x10; ///< `+`: prefix with sign character +const int TIME_FLAG_NO_PAD = 0x20; ///< `-`: do not pad numbers +const int TIME_FLAG_SPACE_PAD = 0x40; ///< `_`: pad numbers with spaces +const int TIME_FLAG_ZERO_PAD = 0x80; ///< `0`: pad numbers with zeros + +// These are the characters allowed in time format codes. +const string TIME_FORMAT_CHARS = "aAbBpPIljwuCyYmdeHkMSfDFRTcxXr%"; + +// Begin time-only constants. It is an error to use these with a duration. +const int TIME_FORMAT_NAME_OF_DAY_ABBR = 0; ///< `%a`: Mon..Sun +const int TIME_FORMAT_NAME_OF_DAY_LONG = 1; ///< `%A`: Monday..Sunday +const int TIME_FORMAT_NAME_OF_MONTH_ABBR = 2; ///< `%b`: Jan..Dec +const int TIME_FORMAT_NAME_OF_MONTH_LONG = 3; ///< `%B`: January..December +const int TIME_FORMAT_AMPM_UPPER = 4; ///< `%p`: AM..PM +const int TIME_FORMAT_AMPM_LOWER = 5; ///< `%P`: am..pm +const int TIME_FORMAT_HOUR_12 = 6; ///< `%I`: 01..12 +const int TIME_FORMAT_HOUR_12_SPACE_PAD = 7; ///< `%l`: alias for %_I +const int TIME_FORMAT_DAY_OF_YEAR = 8; ///< `%j`: 001..336 +const int TIME_FORMAT_DAY_OF_WEEK_0_6 = 9; ///< `%w`: weekdays 0..6 +const int TIME_FORMAT_DAY_OF_WEEK_1_7 = 10; ///< `%u`: weekdays 1..7 +const int TIME_FORMAT_YEAR_CENTURY = 11; ///< `%C`: 0..320 +const int TIME_FORMAT_YEAR_SHORT = 12; ///< `%y`: 00..99 +const int TIME_FORMAT_YEAR_LONG = 13; ///< `%Y`: 0..320000 +const int TIME_FORMAT_MONTH = 14; ///< `%m`: 01..12 +const int TIME_FORMAT_DAY = 15; ///< `%d`: 01..28 +const int TIME_FORMAT_DAY_SPACE_PAD = 16; ///< `%e`: alias for %_d +const int TIME_FORMAT_HOUR_24 = 17; ///< `%H`: 00..23 +const int TIME_FORMAT_HOUR_24_SPACE_PAD = 18; ///< `%k`: alias for %_H +const int TIME_FORMAT_MINUTE = 19; ///< `%M`: 00..59 (depending on conversion factor) +const int TIME_FORMAT_SECOND = 20; ///< `%S`: 00..59 +const int TIME_FORMAT_MILLISECOND = 21; ///< `%f`: 000...999 +const int TIME_FORMAT_DATE_US = 22; ///< `%D`: 06/01/72 +const int TIME_FORMAT_DATE_ISO = 23; ///< `%F`: 1372-06-01 +const int TIME_FORMAT_TIME_US = 24; ///< `%R`: 13:00 +const int TIME_FORMAT_TIME_ISO = 25; ///< `%T`: 13:00:00 +const int TIME_FORMAT_LOCALE_DATETIME = 26; ///< `%c`: locale-specific date and time +const int TIME_FORMAT_LOCALE_DATE = 27; ///< `%x`: locale-specific date +const int TIME_FORMAT_LOCALE_TIME = 28; ///< `%X`: locale-specific time +const int TIME_FORMAT_LOCALE_TIME_AMPM = 29; ///< `%r`: locale-specific AM/PM time +const int TIME_FORMAT_PERCENT = 30; ///< `%%`: % + +// Time format codes with an index less than this number are not valid for +// durations. +const int DURATION_FORMAT_OFFSET = TIME_FORMAT_YEAR_CENTURY; + +// ----- VarNames -------------------------------------------------------------- + +// Prefix for locale names stored on the module to avoid collision +const string LOCALE_PREFIX = "*Locale: "; + +// Stores the default locale on the module +const string LOCALE_DEFAULT = "*DefaultLocale"; + +// Each of these keys stores a CSV list which is evaluated by a format code +const string LOCALE_DAYS = "Days"; // day names (%A) +const string LOCALE_DAYS_ABBR = "DaysAbbr"; // abbreviated day names (%a) +const string LOCALE_MONTHS = "Months"; // month names (%B) +const string LOCALE_MONTHS_ABBR = "MonthsAbbr"; // abbreviated month names (%b) +const string LOCALE_AMPM = "AMPM"; // AM/PM elements (%p and %P) + +// This key stores a CSV list of suffixes used to convert integers to ordinals +// (e.g., 0th, 1st, etc.). +const string LOCALE_ORDINAL_SUFFIXES = "OrdinalSuffixes"; // %On + +// Each of these keys stores a locale-specific format string which is aliased by +// a format code. +const string LOCALE_DATETIME_FORMAT = "DateTimeFormat"; // %c +const string LOCALE_DATE_FORMAT = "DateFormat"; // %x +const string LOCALE_TIME_FORMAT = "TimeFormat"; // %X +const string LOCALE_AMPM_FORMAT = "AMPMFormat"; // %r + +// Each of these keys stores a locale-specific era-based format string which is +// aliased by a format code using the `E` modifier. If no string is stored at +// this key, it will resolve to the non-era based format above. +const string ERA_DATETIME_FORMAT = "EraDateTimeFormat"; // %Ec +const string ERA_DATE_FORMAT = "EraDateFormat"; // %Ex +const string ERA_TIME_FORMAT = "EraTimeFormat"; // %EX + +// Key for Eras json array. Each element of the array is a json object having +// the three keys below. +const string LOCALE_ERAS = "Eras"; + +// Key for era name. Aliased by %EC. +const string ERA_NAME = "Name"; + +// Key for a format string for the year in the era. Aliased by %EY. +const string ERA_YEAR_FORMAT = "YearFormat"; + +// Key for the start of the era. Stored as a date in the form yyyy-mm-dd. +const string ERA_START = "Start"; + +// Key for the number of the year closest to the start date in an era. Used by +// %Ey to display the correct year. For example, if an era starts on 1372-01-01 +// and the current date is 1372-06-01, an offset of 0 would make %Ey display 0, +// while an offset of 1 would make it display 1. +const string ERA_OFFSET = "Offset"; + + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +// ----- Locales --------------------------------------------------------------- + +/// @brief Get the string at a given key in a locale object. +/// @param jLocale A json object containing the locale settings +/// @param sKey The key to return the value of (see the LOCALE_* constants) +/// @param sDefault A default value to return if sKey does not exist in jLocale. +string GetLocaleString(json jLocale, string sKey, string sSuffix = ""); + +/// @brief Set the string at a given key in a locale object. +/// @param jLocale A json object containing the locale settings +/// @param sKey The key to set the value of (see the LOCALE_* constants) +/// @param sValue The value to set the key to +/// @returns The updated locale object +json SetLocaleString(json j, string sKey, string sValue); + +/// @brief Create a new locale object initialized with values from util_c_times. +/// @note If you do not want the default values, use JsonObject() instead. +json NewLocale(); + +/// @brief Get the name of the default locale for the module. +/// @returns The name of the default locale, or the value of DEFAULT_LOCALE from +/// util_c_times.nss if a locale is not set. +string GetDefaultLocale(); + +/// @brief Set the name of the default locale for the module. +/// @param sName The name of the locale (default: DEFAULT_LOCALE) +void SetDefaultLocale(string sName = DEFAULT_LOCALE); + +/// @brief Get a locale object by name. +/// @param sLocale The name of the locale. Will return the default locale if "". +/// @param bInit If TRUE, will return an era with the default values from +/// util_c_times.nss if sLocale does not exist. +/// @returns A json object containing the locale settings, or JsonNull() if no +/// locale named sLocale exists. +json GetLocale(string sLocale = "", int bInit = TRUE); + +/// @brief Save a locale object to a name. +/// @param jLocale A json object containing the locale settings. +/// @param sLocale The name of the locale. Will use the default local if "". +void SetLocale(json jLocale, string sLocale = ""); + +/// @brief Delete a locale by name. +/// @param sLocale The name of the locale. Will use the default local if "". +void DeleteLocale(string sLocale = ""); + +/// @brief Check if a locale exists. +/// @param sLocale The name of the locale. Will use the default local if "". +/// @returns TRUE if sLocale points to a valid json object, other FALSE. +int HasLocale(string sLocale = ""); + +/// @brief Get the name of a month given a locale. +/// @param nMonth The month of the year (1-indexed). +/// @param sMonths A CSV list of 12 month names to search through. If "", will +/// use the month list from a locale. +/// @param sLocale The name of a locale to check for month names if sMonths is +/// "". If sLocale is "", will use the default locale. +/// @returns The name of the month. +string MonthToString(int nMonth, string sMonths = "", string sLocale = ""); + +/// @brief Get the name of a day given a locale. +/// @param nDay The day of the week (1-indexed). +/// @param sDays A CSV list of 7 day names to search through. If "", will use +/// the day list from a locale. +/// @param sLocale The name of a locale to check for day names if sDays is "". +/// If sLocale is "", will use the default locale. +/// @returns The name of the day. +string DayToString(int nDay, string sDays = "", string sLocale = ""); + +// ----- Eras ------------------------------------------------------------------ + +/// @brief Create an era json object. +/// @param sName The name of the era. +/// @param tStart The Time marking the beginning of the era. +/// @param nOffset The number that represents the first year in an era. Used by +/// %Ey to display the correct year. For example, if an era starts on +/// 1372-01-01 and the current date is 1372-06-01, an offset of 0 would make +/// %Ey display 0 while an offset of 1 would make %Ey display 1. The default +/// is 0 since NWN allows year 0. +/// @param sFormat The default format for an era-based year. The format code %EY +/// evaluates to this string for this era. With the default value, the 42nd +/// year of an era named "Foo" would be "4 Foo". +json DefineEra(string sName, struct Time tStart, int nOffset = 0, string sFormat = "%Ey %EC"); + +/// @brief Add an era to a locale. +/// @param jLocale A locale json object. +/// @param jEra An era json object. +/// @returns A modified copy of jLocale with jEra added to its era array. +json AddEra(json jLocale, json jEra); + +/// @brief Get the era in which a time occurs. +/// @param jLocale A locale json object containing an array of eras. +/// @param t A Time to check the era for. +/// @returns A json object for the era in jLocale with the latest start time +/// earlier than t or JsonNull() if no such era is present. +json GetEra(json jLocale, struct Time t); + +/// @brief Get the year of an era given an NWN calendar year. +/// @param jEra A json object matching an era. +/// @param nYear An NWN calendar year (0..32000) +/// @returns The number of the year in the era, or nYear if jEra is not valid. +int GetEraYear(json jEra, int nYear); + +/// @brief Gets a string from an era, falling back to a locale if not set. +/// @param jEra The era to check +/// @param jLocale The locale to fall back to +/// @param sKey The key to get the string from +/// @note If sKey begins with "Era" and was not found on the era or the locale, +/// will check jLocale for sKey without the "Era" prefix. +string GetEraString(json jEra, json jLocale, string sKey); + +// ----- Formatting ------------------------------------------------------------ + +/// @brief Convert an integer into an ordinal number (e.g., 1 -> 1st, 2 -> 2nd). +/// @param n The number to convert. +/// @param sSuffixes A CSV list of suffixes for each integer, starting at 0. If +/// the n <= the length of the list, only the last digit will be checked. If +/// "", will use the suffixes provided by the locale instead. +/// @param sLocale The name of the locale to use when formatting the number. If +/// "", will use the default locale. +string IntToOrdinalString(int n, string sSuffixes = "", string sLocale = ""); + +/// @brief Format a Time into a string. +/// @param t A calendar or duration Time to format. No conversion is performed. +/// @param sFormat A string containing format codes to control the output. +/// @param sLocale The name of the locale to use when formatting the time. If +/// "", will use the default locale. +/// @note See the documentation at the top of this file for the list of possible +/// format codes. +string strftime(struct Time t, string sFormat, string sLocale = ""); + +/// @brief Format a calendar Time into a string. +/// @param t A calendar Time to format. If not a calendar Time, will be +/// converted into one. +/// @param sFormat A string containing format codes to control the output. The +/// default value is equivalent to "%H:%M:%S". +/// @param sLocale The name of the locale to use when formatting the time. If +/// "", will use the default locale. +/// @note This function differs only from FormatTime() in the default value of +/// sFormat. Character codes that apply to calendar Times are still valid. +/// @note See the documentation at the top of this file for the list of possible +/// format codes. +string FormatTime(struct Time t, string sFormat = "%X", string sLocale = ""); + +/// @brief Format a calendar Time into a string. +/// @param t A calendar Time to format. If not a calendar Time, will be +/// converted into one. +/// @param sFormat A string containing format codes to control the output. The +/// default value is equivalent to "%Y-%m-%d". +/// @param sLocale The name of the locale to use when formatting the date. If +/// "", will use the default locale. +/// @note This function differs only from FormatTime() in the default value of +/// sFormat. Character codes that apply to calendar Times are still valid. +/// @note See the documentation at the top of this file for the list of possible +/// format codes. +string FormatDate(struct Time t, string sFormat = "%x", string sLocale = ""); + +/// @brief Format a calendar Time into a string. +/// @param t A calendar Time to format. If not a calendar Time, will be +/// converted into one. +/// @param sFormat A string containing format codes to control the output. The +/// default value is equivalent to "%Y-%m-%d %H:%M:%S:%f". +/// @param sLocale The name of the locale to use when formatting the Time. If +/// "", will use the default locale. +/// @note This function differs only from FormatTime() in the default value of +/// sFormat. Character codes that apply to calendar Times are still valid. +/// @note See the documentation at the top of this file for the list of possible +/// format codes. +string FormatDateTime(struct Time t, string sFormat = "%c", string sLocale = ""); + +/// @brief Format a duration Time into a string. +/// @param t The duration Time to format. If not a duration Time, will be +/// converted into one. +/// @param sFormat A string containing format codes to control the output. The +/// default value is equivalent to ISO 8601 format preceded by the sign of +/// t (`-` if negative, `+` otherwise). +/// @param sLocale The name of the locale to use when formatting the duration. +/// If "", will use the default locale. +/// @note See the documentation at the top of this file for the list of possible +/// format codes. +string FormatDuration(struct Time t, string sFormat = "%+Y-%m-%d %H:%M:%S:%f", string sLocale = ""); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +// ----- Locales --------------------------------------------------------------- + +string GetLocaleString(json jLocale, string sKey, string sDefault = "") +{ + json jElem = JsonObjectGet(jLocale, sKey); + if (JsonGetType(jElem) == JSON_TYPE_STRING && JsonGetString(jElem) != "") + return JsonGetString(jElem); + return sDefault; +} + +json SetLocaleString(json j, string sKey, string sValue) +{ + return JsonObjectSet(j, sKey, JsonString(sValue)); +} + +json NewLocale() +{ + json j = JsonObject(); + j = SetLocaleString(j, LOCALE_ORDINAL_SUFFIXES, DEFAULT_ORDINAL_SUFFIXES); + j = SetLocaleString(j, LOCALE_DAYS, DEFAULT_DAYS); + j = SetLocaleString(j, LOCALE_DAYS_ABBR, DEFAULT_DAYS_ABBR); + j = SetLocaleString(j, LOCALE_MONTHS, DEFAULT_MONTHS); + j = SetLocaleString(j, LOCALE_MONTHS_ABBR, DEFAULT_MONTHS_ABBR); + j = SetLocaleString(j, LOCALE_AMPM, DEFAULT_AMPM); + j = SetLocaleString(j, LOCALE_DATETIME_FORMAT, DEFAULT_DATETIME_FORMAT); + j = SetLocaleString(j, LOCALE_DATE_FORMAT, DEFAULT_DATE_FORMAT); + j = SetLocaleString(j, LOCALE_TIME_FORMAT, DEFAULT_TIME_FORMAT); + j = SetLocaleString(j, LOCALE_AMPM_FORMAT, DEFAULT_AMPM_FORMAT); + + if (DEFAULT_ERA_DATETIME_FORMAT != "") + j = SetLocaleString(j, ERA_DATETIME_FORMAT, DEFAULT_ERA_DATETIME_FORMAT); + + if (DEFAULT_ERA_DATE_FORMAT != "") + j = SetLocaleString(j, ERA_DATE_FORMAT, DEFAULT_ERA_DATE_FORMAT); + + if (DEFAULT_ERA_TIME_FORMAT != "") + j = SetLocaleString(j, ERA_TIME_FORMAT, DEFAULT_ERA_TIME_FORMAT); + + if (DEFAULT_ERA_NAME != "") + j = SetLocaleString(j, ERA_NAME, DEFAULT_ERA_NAME); + + return JsonObjectSet(j, LOCALE_ERAS, JsonArray()); +} + +string GetDefaultLocale() +{ + string sLocale = GetLocalString(GetModule(), LOCALE_DEFAULT); + return sLocale == "" ? DEFAULT_LOCALE : sLocale; +} + +void SetDefaultLocale(string sName = DEFAULT_LOCALE) +{ + SetLocalString(GetModule(), LOCALE_DEFAULT, sName); +} + +json GetLocale(string sLocale = "", int bInit = TRUE) +{ + if (sLocale == "") + sLocale = GetDefaultLocale(); + json j = GetLocalJson(GetModule(), LOCALE_PREFIX + sLocale); + if (bInit && JsonGetType(j) != JSON_TYPE_OBJECT) + j = NewLocale(); + return j; +} + +void SetLocale(json jLocale, string sLocale = "") +{ + if (sLocale == "") + sLocale = GetDefaultLocale(); + SetLocalJson(GetModule(), LOCALE_PREFIX + sLocale, jLocale); +} + +void DeleteLocale(string sLocale = "") +{ + if (sLocale == "") + sLocale = GetDefaultLocale(); + DeleteLocalJson(GetModule(), LOCALE_PREFIX + sLocale); +} + +int HasLocale(string sLocale = "") +{ + return JsonGetType(GetLocale(sLocale, FALSE)) == JSON_TYPE_OBJECT; +} + +string MonthToString(int nMonth, string sMonths = "", string sLocale = "") +{ + if (sMonths == "") + sMonths = GetLocaleString(GetLocale(sLocale), LOCALE_MONTHS); + + return GetListItem(sMonths, (nMonth - 1) % 12); +} + +string DayToString(int nDay, string sDays = "", string sLocale = "") +{ + if (sDays == "") + sDays = GetLocaleString(GetLocale(sLocale), LOCALE_DAYS); + + return GetListItem(sDays, (nDay - 1) % 7); +} + +// ----- Eras ------------------------------------------------------------------ + +json DefineEra(string sName, struct Time tStart, int nOffset = 0, string sFormat = DEFAULT_ERA_YEAR_FORMAT) +{ + json jEra = JsonObject(); + jEra = JsonObjectSet(jEra, ERA_NAME, JsonString(sName)); + jEra = JsonObjectSet(jEra, ERA_YEAR_FORMAT, JsonString(sFormat)); + jEra = JsonObjectSet(jEra, ERA_START, TimeToJson(tStart)); + return JsonObjectSet(jEra, ERA_OFFSET, JsonInt(nOffset)); +} + +json AddEra(json jLocale, json jEra) +{ + json jEras = JsonObjectGet(jLocale, LOCALE_ERAS); + if (JsonGetType(jEras) != JSON_TYPE_ARRAY) + jEras = JsonArray(); + + jEras = JsonArrayInsert(jEras, jEra); + return JsonObjectSet(jLocale, LOCALE_ERAS, jEras); +} + +json GetEra(json jLocale, struct Time t) +{ + if (t.Type == TIME_TYPE_DURATION) + return JsonNull(); + + json jEras = JsonObjectGet(jLocale, LOCALE_ERAS); + json jEra; // The closest era to the Time + struct Time tEra; // The start Time of jEra + int i, nLength = JsonGetLength(jEras); + + for (i = 0; i < nLength; i++) + { + json jCmp = JsonArrayGet(jEras, i); + struct Time tCmp = JsonToTime(JsonObjectGet(jCmp, ERA_START)); + switch (CompareTime(t, tCmp)) + { + case 0: return jCmp; + case 1: + { + if (CompareTime(tCmp, tEra) >= 0) + { + tEra = tCmp; + jEra = jCmp; + } + } + } + } + + return jEra; +} + +int GetEraYear(json jEra, int nYear) +{ + int nOffset = JsonGetInt(JsonObjectGet(jEra, ERA_OFFSET)); + struct Time tStart = JsonToTime(JsonObjectGet(jEra, ERA_START)); + return nYear - tStart.Year + nOffset; +} + +string GetEraString(json jEra, json jLocale, string sKey) +{ + json jValue = JsonObjectGet(jEra, sKey); + if (JsonGetType(jValue) != JSON_TYPE_STRING) + { + jValue = JsonObjectGet(jLocale, sKey); + if (JsonGetType(jValue) != JSON_TYPE_STRING && + (GetStringSlice(sKey, 0, 2) == "Era")) + jValue = JsonObjectGet(jLocale, GetStringSlice(sKey, 3)); + } + + return JsonGetString(jValue); +} + +// ----- Formatting ------------------------------------------------------------ + +string IntToOrdinalString(int n, string sSuffixes = "", string sLocale = "") +{ + if (sSuffixes == "") + { + json jLocale = GetLocale(sLocale); + sSuffixes = GetLocaleString(jLocale, LOCALE_ORDINAL_SUFFIXES, DEFAULT_ORDINAL_SUFFIXES); + } + + int nIndex = abs(n) % 100; + if (nIndex >= CountList(sSuffixes)) + nIndex = abs(n) % 10; + + return IntToString(n) + GetListItem(sSuffixes, nIndex); +} + +string strftime(struct Time t, string sFormat, string sLocale) +{ + int nOffset, nPos; + int nSign = GetTimeSign(t); + json jValues = JsonArray(); + json jLocale = GetLocale(sLocale); + json jEra = GetEra(jLocale, t); + string sOrdinals = GetLocaleString(jLocale, LOCALE_ORDINAL_SUFFIXES, DEFAULT_ORDINAL_SUFFIXES); + int nDigitsIndex = log2(TIME_FLAG_ZERO_PAD); + + while ((nPos = FindSubString(sFormat, "%", nOffset)) != -1) + { + nOffset = nPos; + + // Check for flags + int nFlag, nFlags; + string sPadding, sWidth, sChar; + + while ((nFlag = FindSubString(TIME_FLAG_CHARS, (sChar = GetChar(sFormat, ++nPos)))) != -1) + { + // If this character is not a digit after 0, we create a flag for it + // and add it to our list of flags. + if (nFlag < nDigitsIndex) + nFlags |= (1 << nFlag); + else + { + // The user has specified a width for the item. Parse all the + // numbers. + sWidth = ""; // in case the user added a width twice and separated with another flag. + while (GetIsNumeric(sChar)) + { + sWidth += sChar; + sChar = GetChar(sFormat, ++nPos); + } + + nPos--; + } + } + + string sValue; + int nValue; + int bAllowEmpty; + int nPadding = 2; // Most numeric formats use this + + // We offset where we start looking for format codes based on whether + // this is a calendar Time or duration Time. Durations cannot use time + // codes that only make sense in the context of a calendar Time. + int nFormat = FindSubString(TIME_FORMAT_CHARS, sChar, t.Type ? 0 : DURATION_FORMAT_OFFSET); + switch (nFormat) + { + case -1: + { + string sError = GetStringSlice(sFormat, nOffset, nPos); + string sColored = GetStringSlice(sFormat, 0, nOffset - 1) + + HexColorString(sError, COLOR_RED) + + GetStringSlice(sFormat, nPos + 1); + Error("Illegal time format \"" + sError + "\": " + sColored); + sFormat = ReplaceSubString(sFormat, "%" + sError, nOffset, nPos); + continue; + } + + // Note that some of these are meant to fall through + case TIME_FORMAT_DAY_SPACE_PAD: // %e + sPadding = " "; + case TIME_FORMAT_DAY: // %d + nValue = t.Day; + break; + case TIME_FORMAT_HOUR_24_SPACE_PAD: // %H + sPadding = " "; + case TIME_FORMAT_HOUR_24: // %H + nValue = t.Hour; + break; + case TIME_FORMAT_HOUR_12_SPACE_PAD: // %l + sPadding = " "; + case TIME_FORMAT_HOUR_12: // %I + nValue = t.Hour > 12 ? t.Hour % 12 : t.Hour; + nValue = nValue ? nValue : 12; + break; + case TIME_FORMAT_MONTH: // %m + nValue = t.Month; + break; + case TIME_FORMAT_MINUTE: // %M + nValue = t.Minute; + break; + case TIME_FORMAT_SECOND: // %S + nValue = t.Second; + break; + case TIME_FORMAT_MILLISECOND: // %f + nValue = t.Millisecond; + nPadding = 3; + break; + case TIME_FORMAT_DAY_OF_YEAR: // %j + nValue = t.Month * 28 + t.Day; + nPadding = 3; + break; + case TIME_FORMAT_DAY_OF_WEEK_0_6: // %w + nValue = t.Day % 7; + nPadding = 1; + break; + case TIME_FORMAT_DAY_OF_WEEK_1_7: // %u + nValue = (t.Day % 7) + 1; + nPadding = 1; + break; + case TIME_FORMAT_AMPM_UPPER: // %p + case TIME_FORMAT_AMPM_LOWER: // %P + bAllowEmpty = TRUE; + sValue = GetLocaleString(jLocale, LOCALE_AMPM); + sValue = GetListItem(sValue, t.Hour % 24 >= 12); + if (nFormat == TIME_FORMAT_AMPM_LOWER) + sValue = GetStringLowerCase(sValue); + break; + case TIME_FORMAT_NAME_OF_DAY_LONG: // %A + bAllowEmpty = TRUE; + sValue = GetLocaleString(jLocale, LOCALE_DAYS); + sValue = DayToString(t.Day, sValue); + break; + case TIME_FORMAT_NAME_OF_DAY_ABBR: // %a + bAllowEmpty = TRUE; + sValue = GetLocaleString(jLocale, LOCALE_DAYS); + sValue = GetLocaleString(jLocale, LOCALE_DAYS_ABBR, sValue); + sValue = DayToString(t.Day, sValue); + break; + case TIME_FORMAT_NAME_OF_MONTH_LONG: // %B + bAllowEmpty = TRUE; + sValue = GetLocaleString(jLocale, LOCALE_MONTHS); + sValue = MonthToString(t.Month, sValue); + break; + case TIME_FORMAT_NAME_OF_MONTH_ABBR: // %b + bAllowEmpty = TRUE; + sValue = GetLocaleString(jLocale, LOCALE_MONTHS); + sValue = GetLocaleString(jLocale, LOCALE_MONTHS_ABBR, sValue); + sValue = MonthToString(t.Month, sValue); + break; + + // We handle literal % here instead of replacing it directly because + // we want the user to be able to pad it if desired. + case TIME_FORMAT_PERCENT: // %% + sValue = "%"; + break; + + case TIME_FORMAT_YEAR_CENTURY: // %C, %EC + if (nFlags & TIME_FLAG_ERA) + sValue = GetEraString(jEra, jLocale, ERA_NAME); + nValue = t.Year / 100; + break; + case TIME_FORMAT_YEAR_SHORT: // %y, %Ey + nValue = (nFlags & TIME_FLAG_ERA) ? GetEraYear(jEra, t.Year) : t.Year % 100; + break; + + case TIME_FORMAT_YEAR_LONG: // %Y, %EY + if (nFlags & TIME_FLAG_ERA) + { + sValue = GetEraString(jEra, jLocale, ERA_YEAR_FORMAT); + if (sValue != "") + { + sFormat = ReplaceSubString(sFormat, sValue, nOffset, nPos); + continue; + } + } + + nValue = t.Year; + nPadding = 4; + break; + + // These codes are shortcuts to common operations. We replace the + // parsed code with the substitution and re-parse from the same + // offset. + case TIME_FORMAT_DATE_US: // %D + sFormat = ReplaceSubString(sFormat, "%m/%d/%y", nOffset, nPos); + continue; + case TIME_FORMAT_DATE_ISO: // %F + sFormat = ReplaceSubString(sFormat, "%Y-%m-%d", nOffset, nPos); + continue; + case TIME_FORMAT_TIME_US: // %R + sFormat = ReplaceSubString(sFormat, "%H:%M", nOffset, nPos); + continue; + case TIME_FORMAT_TIME_ISO: // %T + sFormat = ReplaceSubString(sFormat, "%H:%M:%S", nOffset, nPos); + continue; + case TIME_FORMAT_LOCALE_DATETIME: // %c, %Ec + if (nFlags & TIME_FLAG_ERA) + sValue = GetEraString(jEra, jLocale, ERA_DATETIME_FORMAT); + else + sValue = GetLocaleString(jLocale, LOCALE_DATETIME_FORMAT, DEFAULT_DATETIME_FORMAT); + sFormat = ReplaceSubString(sFormat, sValue, nOffset, nPos); + continue; + case TIME_FORMAT_LOCALE_DATE: // %x, %Ex + if (nFlags & TIME_FLAG_ERA) + sValue = GetEraString(jEra, jLocale, ERA_DATE_FORMAT); + else + sValue = GetLocaleString(jLocale, LOCALE_DATE_FORMAT, DEFAULT_DATE_FORMAT); + sFormat = ReplaceSubString(sFormat, sValue, nOffset, nPos); + continue; + case TIME_FORMAT_LOCALE_TIME: // %c, %Ec + if (nFlags & TIME_FLAG_ERA) + sValue = GetEraString(jEra, jLocale, ERA_TIME_FORMAT); + else + sValue = GetLocaleString(jLocale, LOCALE_TIME_FORMAT, DEFAULT_TIME_FORMAT); + sFormat = ReplaceSubString(sFormat, sValue, nOffset, nPos); + continue; + case TIME_FORMAT_LOCALE_TIME_AMPM: // %r + sValue = GetLocaleString(jLocale, LOCALE_AMPM_FORMAT, DEFAULT_AMPM_FORMAT); + sFormat = ReplaceSubString(sFormat, sValue, nOffset, nPos); + continue; + } + + if ((sValue == "" && !bAllowEmpty) && (nFlags & TIME_FLAG_ORDINAL)) + sValue = IntToOrdinalString(nValue, sOrdinals); + + if (nFlags & TIME_FLAG_NO_PAD) + sPadding = ""; + else if (sValue != "" || bAllowEmpty) + sPadding = " " + sWidth; + else + { + if (nFlags & TIME_FLAG_SPACE_PAD) + sPadding = " "; + else if (nFlags & TIME_FLAG_ZERO_PAD || sPadding == "") + sPadding = "0"; + + sPadding += sWidth != "" ? sWidth : IntToString(nPadding); + } + + if (sValue != "" || bAllowEmpty) + { + if (nFlags & TIME_FLAG_UPPERCASE) + sValue = GetStringUpperCase(sValue); + jValues = JsonArrayInsert(jValues, JsonString(sValue)); + sFormat = ReplaceSubString(sFormat, "%" + sPadding + "s", nOffset, nPos); + } + else + { + if (nFlags & TIME_FLAG_SIGN) + sValue = nSign < 0 ? "-" : "+"; + + if (nFlags & TIME_FLAG_COMMAS) + sPadding = "," + sPadding; + + jValues = JsonArrayInsert(jValues, JsonInt(abs(nValue))); + sFormat = ReplaceSubString(sFormat, sValue + "%" + sPadding + "d", nOffset, nPos); + } + + // Continue parsing from the end of the format string + nOffset = nPos + GetStringLength(sPadding); + } + + // Interpolate the values + return FormatValues(jValues, sFormat); +} + +string FormatTime(struct Time t, string sFormat = "%X", string sLocale = "") +{ + return strftime(DurationToTime(t), sFormat, sLocale); +} + +string FormatDate(struct Time t, string sFormat = "%x", string sLocale = "") +{ + return strftime(DurationToTime(t), sFormat, sLocale); +} + +string FormatDateTime(struct Time t, string sFormat = "%c", string sLocale = "") +{ + return strftime(DurationToTime(t), sFormat, sLocale); +} + +string FormatDuration(struct Time t, string sFormat = "%+Y-%m-%d %H:%M:%S:%f", string sLocale = "") +{ + return strftime(TimeToDuration(t), sFormat, sLocale); +} diff --git a/_module/nss/util_i_strings.nss b/_module/nss/util_i_strings.nss new file mode 100644 index 0000000..5b48116 --- /dev/null +++ b/_module/nss/util_i_strings.nss @@ -0,0 +1,422 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_strings.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions for manipulating strings. +/// ---------------------------------------------------------------------------- +/// @details This file holds utility functions for manipulating strings. +/// ---------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string CHARSET_NUMERIC = "0123456789"; +const string CHARSET_ALPHA = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; +const string CHARSET_ALPHA_LOWER = "abcdefghijklmnopqrstuvwxyz"; +const string CHARSET_ALPHA_UPPER = "ABCDEFGHIJKLMNOPQRSTUVWXYZ"; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Return the number of occurrences of a substring within a string. +/// @param sString The string to search. +/// @param sSubString The substring to search for. +int GetSubStringCount(string sString, string sSubString); + +/// @brief Return the position of a given occurrence of a substring within a +/// string. +/// @param sString The string to search. +/// @param sSubString The substring to search for. +/// @param nNth The occurrence to search for. Uses a zero-based index. +/// @returns The position of the start of the nNth occurrence of the substring, +/// or -1 if the substring did not occur at least nNth + 1 times. +int FindSubStringN(string sString, string sSubString, int nNth = 0); + +/// @brief Return the character at a position in a string. +/// @param sString The string to search. +/// @param nPos The position to check. +/// @returns "" if sString is not nPos + 1 characters long. +string GetChar(string sString, int nPos); + +/// @brief Return the substring of a string bounded by a start and end position. +/// @param sString The string to search. +/// @param nStart The starting position of the substring to return. +/// @param nEnd The ending position of the substring to return. If -1, will +/// return to the end of the string. +/// @returns "" if nStart is not at least nStart + 1 characters long or if nEnd +/// is < nStart and not -1. +/// @note Both nStart and nEnd are inclusive, so if nStart == nEnd, the +/// character at that index will be returned. +string GetStringSlice(string sString, int nStart, int nEnd = -1); + +/// @brief Replace the substring bounded by a string slice with another string. +/// @param sString The string to search. +/// @param sSub The substring to replace with. +/// @param nStart The starting position in sString of the substring to replace. +/// @param nEnd The ending position in sString of the substring to replace. +string ReplaceSubString(string sString, string sSub, int nStart, int nEnd); + +/// @brief Replace a substring in a string with another string. +/// @param sString The string to search. +/// @param sToken The substring to search for. +/// @param sSub The substring to replace with. +string SubstituteSubString(string sString, string sToken, string sSub); + +/// @brief Replace all substrings in a string with another string. +/// @param sString The string to search. +/// @param sToken The substring to search for. +/// @param sSub The substring to replace with. +string SubstituteSubStrings(string sString, string sToken, string sSub); + +/// @brief Return whether a string contains a substring. +/// @param sString The string to search. +/// @param sSubString The substring to search for. +/// @param nStart The position in sString to begin searching from (0-based). +/// @returns TRUE if sSubString is in sString, FALSE otherwise. +int HasSubString(string sString, string sSubString, int nStart = 0); + +/// @brief Return whether any of a string's characters are in a character set. +/// @param sString The string to search. +/// @param sSet The set of characters to search for. +/// @returns TRUE if any characters are in the set; FALSE otherwise. +int GetAnyCharsInSet(string sString, string sSet); + +/// @brief Return whether all of a string's characters are in a character set. +/// @param sString The string to search. +/// @param sSet The set of characters to search for. +/// @returns TRUE if all characters are in the set; FALSE otherwise. +int GetAllCharsInSet(string sString, string sSet); + +/// @brief Return whether all letters in a string are upper-case. +/// @param sString The string to check. +int GetIsUpperCase(string sString); + +/// @brief Return whether all letters in a string are lower-case. +/// @param sString The string to check. +int GetIsLowerCase(string sString); + +/// @brief Return whether all characters in sString are letters. +/// @param sString The string to check. +int GetIsAlpha(string sString); + +/// @brief Return whether all characters in sString are digits. +/// @param sString The string to check. +int GetIsNumeric(string sString); + +/// @brief Return whether all characters in sString are letters or digits. +/// @param sString The string to check. +int GetIsAlphaNumeric(string sString); + +/// @brief Trim characters from the left side of a string. +/// @param sString The string to trim. +/// @param sRemove The set of characters to remove. +string TrimStringLeft(string sString, string sRemove = " "); + +/// @brief Trim characters from the right side of a string. +/// @param sString The string to trim. +/// @param sRemove The set of characters to remove. +string TrimStringRight(string sString, string sRemove = " "); + +/// @brief Trim characters from both sides of a string. +/// @param sString The string to trim. +/// @param sRemove The set of characters to remove. +string TrimString(string sString, string sRemove = " "); + +/// @brief Interpolate values from a json array into a string using sqlite's +/// printf(). +/// @param jArray A json array containing float, int, or string elements to +/// interpolate. The number of elements must match the number of format +/// specifiers in sFormat. +/// @param sFormat The string to interpolate the values into. Must contain +/// format specifiers that correspond to the elements in jArray. For details +/// on format specifiers, see https://sqlite.org/printf.html. +/// @example +/// FormatValues(JsonParse("[\"Blue\", 255]"), "%s: #%06X"); // "Blue: #0000FF" +string FormatValues(json jArray, string sFormat); + +/// @brief Interpolate a float into a string using sqlite's printf(). +/// @param f A float to interpolate. Will be passed as an argument to the query +/// as many times as necessary to cover all format specifiers. +/// @param sFormat The string to interpolate the value into. For details on +/// format specifiers, see https://sqlite.org/printf.html. +/// @example +/// FormatFloat(15.0, "%d"); // "15" +/// FormatFloat(15.0, "%.2f"); // "15.00" +/// FormatFloat(15.0, "%05.1f"); // "015.0" +string FormatFloat(float f, string sFormat); + +/// @brief Interpolate an int into a string using sqlite's printf(). +/// @param n An int to interpolate. Will be passed as an argument to the query +/// as many times as necessary to cover all format specifiers. +/// @param sFormat The string to interpolate the value into. For details on +/// format specifiers, see https://sqlite.org/printf.html. +/// @example +/// FormatInt(15, "%d"); // "15" +/// FormatInt(15, "%04d"); // "0015" +/// FormatInt(15, "In hexadecimal, %d is %#x"); // "In hexadecimal, 15 is 0xf" +/// FormatInt(1000, "%,d"); // "1,000" +string FormatInt(int n, string sFormat); + +/// @brief Interpolate a string into another string using sqlite's printf(). +/// @param s A string to interpolate. Will be passed as an argument to the query +/// as many times as necessary to cover all format specifiers. +/// @param sFormat The string to interpolate the value into. For details on +/// format specifiers, see https://sqlite.org/printf.html. +/// @example +/// FormatString("foo", "%sbar"); // "foobar" +/// FormatString("foo", "%5sbar"); // " foobar" +/// FormatString("foo", "%-5sbar"); // "foo bar" +string FormatString(string s, string sFormat); + +/// @brief Substitute tokens in a string with values from a json array. +/// @param s The string to interpolate the values into. Should have tokens wich +/// contain sDesignator followed by a number denoting the position of the +/// value in jArray (1-based index). +/// @param jArray An array of values to interpolate. May be any combination of +/// strings, floats, decimals, or booleans. +/// @param sDesignator The character denoting the beginning of a token. +/// @example +/// // Assumes jArray = ["Today", 34, 2.5299999999, true]; +/// SubstituteString("$1, I ran $2 miles.", jArray); // "Today, I ran 34 miles." +/// SubstituteString("The circle's radius is $3.", jArray); // "The circle's radius is 2.53." +/// SubstituteString("The applicant answered: $4", jArray); // "The applicant answered: true" +string SubstituteString(string s, json jArray, string sDesignator = "$"); + +/// @brief Repeats a string multiple times. +/// @param s The string to repeat. +/// @param n The number of times to repeat s. +/// @returns The repeated string. +string RepeatString(string s, int n); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +int GetSubStringCount(string sString, string sSubString) +{ + if (sString == "" || sSubString == "") + return 0; + + int nLength = GetStringLength(sSubString); + int nCount, nPos = FindSubString(sString, sSubString); + + while (nPos != -1) + { + nCount++; + nPos = FindSubString(sString, sSubString, nPos + nLength); + } + + return nCount; +} + +int FindSubStringN(string sString, string sSubString, int nNth = 0) +{ + if (nNth < 0 || sString == "" || sSubString == "") + return -1; + + int nLength = GetStringLength(sSubString); + int nPos = FindSubString(sString, sSubString); + + while (--nNth >= 0 && nPos != -1) + nPos = FindSubString(sString, sSubString, nPos + nLength); + + return nPos; +} + +string GetChar(string sString, int nPos) +{ + return GetSubString(sString, nPos, 1); +} + +string GetStringSlice(string sString, int nStart, int nEnd = -1) +{ + int nLength = GetStringLength(sString); + if (nEnd < 0 || nEnd > nLength) + nEnd = nLength; + + if (nStart < 0 || nStart > nEnd) + return ""; + + return GetSubString(sString, nStart, nEnd - nStart + 1); +} + +string ReplaceSubString(string sString, string sSub, int nStart, int nEnd) +{ + int nLength = GetStringLength(sString); + if (nStart < 0 || nStart >= nLength || nStart > nEnd) + return sString; + + return GetSubString(sString, 0, nStart) + sSub + + GetSubString(sString, nEnd + 1, nLength - nEnd); +} + +string SubstituteSubString(string sString, string sToken, string sSub) +{ + int nPos; + if ((nPos = FindSubString(sString, sToken)) == -1) + return sString; + + return ReplaceSubString(sString, sSub, nPos, nPos + GetStringLength(sToken) - 1); +} + +string SubstituteSubStrings(string sString, string sToken, string sSub) +{ + while (FindSubString(sString, sToken) >= 0) + sString = SubstituteSubString(sString, sToken, sSub); + + return sString; +} + +int HasSubString(string sString, string sSubString, int nStart = 0) +{ + return FindSubString(sString, sSubString, nStart) >= 0; +} + +int GetAnyCharsInSet(string sString, string sSet) +{ + int i, nLength = GetStringLength(sString); + for (i = 0; i < nLength; i++) + { + if (HasSubString(sSet, GetChar(sString, i))) + return TRUE; + } + return FALSE; +} + +int GetAllCharsInSet(string sString, string sSet) +{ + int i, nLength = GetStringLength(sString); + for (i = 0; i < nLength; i++) + { + if (!HasSubString(sSet, GetChar(sString, i))) + return FALSE; + } + return TRUE; +} + +int GetIsUpperCase(string sString) +{ + return GetAllCharsInSet(sString, CHARSET_ALPHA_UPPER + CHARSET_NUMERIC); +} + +int GetIsLowerCase(string sString) +{ + return GetAllCharsInSet(sString, CHARSET_ALPHA_LOWER + CHARSET_NUMERIC); +} + +int GetIsAlpha(string sString) +{ + return GetAllCharsInSet(sString, CHARSET_ALPHA); +} + +int GetIsNumeric(string sString) +{ + return GetAllCharsInSet(sString, CHARSET_NUMERIC); +} + +int GetIsAlphaNumeric(string sString) +{ + return GetAllCharsInSet(sString, CHARSET_ALPHA + CHARSET_NUMERIC); +} + +string TrimStringLeft(string sString, string sRemove = " ") +{ + return RegExpReplace("^(?:" + sRemove + ")*", sString, ""); +} + +string TrimStringRight(string sString, string sRemove = " ") +{ + return RegExpReplace("(:?" + sRemove + ")*$", sString, ""); +} + +string TrimString(string sString, string sRemove = " ") +{ + return RegExpReplace("^(:?" + sRemove + ")*|(?:" + sRemove + ")*$", sString, ""); +} + +string FormatValues(json jArray, string sFormat) +{ + if (JsonGetType(jArray) != JSON_TYPE_ARRAY) + return ""; + + string sArgs; + int i, nLength = JsonGetLength(jArray); + for (i = 0; i < nLength; i++) + sArgs += ", @" + IntToString(i); + + sqlquery q = SqlPrepareQueryObject(GetModule(), "SELECT printf(@format" + sArgs + ");"); + SqlBindString(q, "@format", sFormat); + for (i = 0; i < nLength; i++) + { + string sParam = "@" + IntToString(i); + json jValue = JsonArrayGet(jArray, i); + switch (JsonGetType(jValue)) + { + case JSON_TYPE_FLOAT: SqlBindFloat (q, sParam, JsonGetFloat (jValue)); break; + case JSON_TYPE_INTEGER: SqlBindInt (q, sParam, JsonGetInt (jValue)); break; + case JSON_TYPE_STRING: SqlBindString(q, sParam, JsonGetString(jValue)); break; + default: break; + } + } + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +string FormatFloat(float f, string sFormat) +{ + json jArray = JsonArray(); + int i, nCount = GetSubStringCount(sFormat, "%"); + for (i = 0; i < nCount; i++) + JsonArrayInsertInplace(jArray, JsonFloat(f)); + return FormatValues(jArray, sFormat); +} + +string FormatInt(int n, string sFormat) +{ + json jArray = JsonArray(); + int i, nCount = GetSubStringCount(sFormat, "%"); + for (i = 0; i < nCount; i++) + JsonArrayInsertInplace(jArray, JsonInt(n)); + return FormatValues(jArray, sFormat); +} + +string FormatString(string s, string sFormat) +{ + json jArray = JsonArray(); + int i, nCount = GetSubStringCount(sFormat, "%"); + for (i = 0; i < nCount; i++) + JsonArrayInsertInplace(jArray, JsonString(s)); + return FormatValues(jArray, sFormat); +} + +string SubstituteString(string s, json jArray, string sDesignator = "$") +{ + if (JsonGetType(jArray) != JSON_TYPE_ARRAY) + return s; + + int n; for (n = JsonGetLength(jArray) - 1; n >= 0; n--) + { + string sValue; + json jValue = JsonArrayGet(jArray, n); + int nType = JsonGetType(jValue); + if (nType == JSON_TYPE_STRING) sValue = JsonGetString(jValue); + else if (nType == JSON_TYPE_INTEGER) sValue = IntToString(JsonGetInt(jValue)); + else if (nType == JSON_TYPE_FLOAT) sValue = FormatFloat(JsonGetFloat(jValue), "%!f"); + else if (nType == JSON_TYPE_BOOL) sValue = JsonGetInt(jValue) == 1 ? "true" : "false"; + else continue; + + s = SubstituteSubStrings(s, sDesignator + IntToString(n + 1), sValue); + } + + return s; +} + +string RepeatString(string s, int n) +{ + string sResult; + while (n-- > 0) + sResult += s; + + return sResult; +} diff --git a/_module/nss/util_i_targeting.nss b/_module/nss/util_i_targeting.nss new file mode 100644 index 0000000..e5e1650 --- /dev/null +++ b/_module/nss/util_i_targeting.nss @@ -0,0 +1,908 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_targeting.nss +/// @author Ed Burke (tinygiant98) +/// @brief Functions for managing forced targeting. +/// ---------------------------------------------------------------------------- +/// @details +/* +This system is designed to take advantage of NWN:EE's ability to forcibly enter +Targeting Mode for any given PC. It is designed to add a single-use, multi-use, +or unlimited-use hook to the specified PC. Once the PC has satisfied the +conditions of the hook, or manually exited targeting mode, the targeted +objects/locations will be saved and a specified script will be run. + +## Setup + +1. You must attach a targeting event script to the module. For example, in your +module load script, you can add this line: + + SetEventScript(GetModule(), EVENT_SCRIPT_MODULE_ON_PLAYER_TARGET, "module_opt"); + +where "module_opt" is the script that will handle all forced targeting. + +2. The chosen script ("module_opt") must contain reference to the +util_i_targeting function SatisfyTargetingHook(). An example of this follows. + +```nwscript +#include "util_i_targeting" + +void main() +{ + object oPC = GetLastPlayerToSelectTarget(); + + if (SatisfyTargetingHook(oPC)) + { + // This PC was marked as a targeter, do something here. + } +} +``` + +Alternately, if you want the assigned targeting hook scripts to handle +everything, you can just let the system know a targeting event happened: + +```nwscript +void main() +{ + object oPC = GetLastPlayerToSelectTarget(); + SatisfyTargetingHook(oPC); +} +``` + +If oPC didn't have a targeting hook specified, nothing happens. + +## Usage + +The design of this system centers around a module-wide list of "Targeting Hooks" +that are accessed by util_i_targeting when a player targets any object or +manually exits targeting mode. These hooks are stored in the module's organic +sqlite database. All targeting hook information is volatile and will be reset +when the server/module is reset. + +This is the prototype for the `AddTargetingHook()` function: + +```nwscript +int AddTargetingHook(object oPC, string sVarName, int nObjectType = OBJECT_TYPE_ALL, string sScript = "", int nUses = 1); +``` + +- `oPC` is the PC object that will be associated with this hook. This PC will be + the player that will be entered into Targeting Mode. Additionally, the results + of his targeting will be saved to the PC object. +- `sVarName` is the variable name to save the results of targeting to. This + allows for targeting hooks to be added that can be saved to different + variables for several purposes. +- `nObjectType` is the limiting variable for the types of objects the PC can + target when they are in targeting mode forced by this hook. It is an optional + parameter and can be bitmasked with any visible `OBJECT_TYPE_*` constant. +- `sScript` is the resref of the script that will run once the targeting + conditions have been satisfied. For example, if you create a multi-use + targeting hook, this script will run after all uses have been exhausted. This + script will also run if the player manually exits targeting mode without + selecting a target. Optional. A script-run is not always desirable. The + targeted object may be required for later use, so a script entry is not a + requirement. +- `nUses` is the number of times this target hook can be used before it is + deleted. This is designed to allow multiple targets to be selected and saved + to the same variable name sVarName. Multi-selection could be useful for DMs in + defining DM Experience members, even from different parties, or selecting + multiple NPCs to accomplish a specific action. Optional, defaulted to 1. + + Note: Targeting mode uses specified by `nUses` will be decremented every time + a player selects a target. Uses will also be decremented when a user manually + exits targeting mode. Manually exiting targeting mode will delete the + targeting hook, but any selected targets before exiting targeting mode will be + saved to the specified variable. + +To add a single-use targeting hook that enters the PC into targeting mode, allows +for the selection of a single placeable | creature, then runs the script +"temp_target" upon exiting target mode or selecting a single target: + +```nwscript +int nObjectType = OBJECT_TYPE_PLACEABLE | OBJECT_TYPE_CREATURE; +AddTargetingHook(oPC, "spell_target", nObjectType, "temp_target"); +``` + +To add a multi-use targeting hook that enters the PC into targeting mode, allows +for the selection of a specified number of placeables | creatures, then runs the +script "DM_Party" upon exiting targeting mode or selecting the specified number +of targets: + +```nwscript +int nObjectType = OBJECT_TYPE_PLACEABLE | OBJECT_TYPE_CREATURE; +AddTargetingHook(oPC, "DM_Party", nObjectType, "DM_Party", 3); +``` + +> Note: In this case, the player can select up to three targets to save to the + "DM_Party" variable. + +To add an unlmited-use targeting hook that enters the PC into targeting mode, +allows for the selection of an unspecified number of creatures, then runs the +script "temp_target" upon exiting targeting mode or selection of an invalid +target: + +```nwscript +int nObjectType = OBJECT_TYPE_CREATURE; +AddTargetingHook(oPC, "NPC_Townspeople", nObjectType, "temp_target", -1); +``` + +Here is an example "temp_target" post-targeting script that will access each of +the targets saved to the specified variable and send their data to the chat log: + +```nwscript +#include "util_i_targeting" + +void main() +{ + object oPC = OBJECT_SELF; + int n, nCount = CountTargetingHookTargets(oPC, "NPC_Townspeople"); + + for (n = 0; n < nCount; n++) + { + object oTarget = GetTargetingHookObject(oPC, "NPC_Townspeople", n); + location lTarget = GetTargetingHookLocation(oPC, "NPC_Townspeople", n); + vector vTarget = GetTargetingHookPosition(oPC, "NPC_Townspeople", n); + } +} +``` + +Note: Target objects and positions saved to the variables are persistent while +the server is running, but are not persistent (though they can be made so). If +you wish to overwrite a set of target data with a variable you've already used, +ensure you first delete the current target data with the function +`DeleteTargetingHookTargets();`. +*/ + +#include "util_c_targeting" +#include "util_i_debug" +#include "util_i_varlists" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// VarList names for the global targeting hook lists +const string TARGET_HOOK_ID = "TARGET_HOOK_ID"; +const string TARGET_HOOK_BEHAVIOR = "TARGET_HOOK_BEHAVIOR"; + +// List Behaviors +const int TARGET_BEHAVIOR_ADD = 1; +const int TARGET_BEHAVIOR_DELETE = 2; + +// Targeting Hook Data Structure +struct TargetingHook +{ + int nHookID; + int nObjectType; + int nUses; + object oPC; + string sVarName; + string sScript; + int nValidCursor; + int nInvalidCursor; +}; + +struct TargetingHook TARGETING_HOOK_INVALID; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Creates targeting hook data tables in the module's sqlite database. +/// @param bReset If TRUE, attempts to drop the tables before creation. +void CreateTargetingDataTables(int bReset = FALSE); + +/// @brief Retrieve targeting hook data. +/// @param nHookID The targeting hook's ID. +/// @returns A TargetingHook containing all targeting hook data associated with +/// nHookID. +struct TargetingHook GetTargetingHookDataByHookID(int nHookID); + +/// @brief Retrieve targeting hook data. +/// @param oPC The PC object associated with the targeting hook. +/// @param sVarName The varname associated with the targeting hook. +/// @returns A TargetingHook containing all targeting hook data associated with +/// nHookID. +struct TargetingHook GetTargetingHookDataByVarName(object oPC, string sVarName); + +/// @brief Retrieve a list of targets. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The index of the target to retrieve from the list. If omitted, +/// the entire target list will be returned. +/// @returns A prepared sqlquery containing the target list associated with +/// oPC's sVarName. +sqlquery GetTargetList(object oPC, string sVarName, int nIndex = -1); + +/// @brief Add a target to a target list. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param oTarget The target object to be added to the target list. +/// @param oArea The area object where oTarget is located. +/// @param vTarget The position of oTarget within oArea. +/// @returns The number of targets on oPC's target list sVarName after insertion. +int AddTargetToTargetList(object oPC, string sVarName, object oTarget, object oArea, vector vTarget); + +/// @brief Delete oPC's sVarName target list. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +void DeleteTargetList(object oPC, string sVarName); + +/// @brief Delete a targeting hook and all associated targeting hook data. +/// @param nHookID The targeting hook's ID. +void DeleteTargetingHook(int nHookID); + +/// @brief Force the PC object associated with targeting hook nHookID to enter +/// targeting mode using properties set by AddTargetingHook(). +/// @param nHookID The targeting hook's ID. +/// @param nBehavior The behavior desired from the targeting session. Must be +/// a TARGET_BEHAVIOR_* constant. +void EnterTargetingModeByHookID(int nHookID, int nBehavior = TARGET_BEHAVIOR_ADD); + +/// @brief Force the PC object associated with targeting hook nHookID to enter +/// targeting mode using properties set by AddTargetingHook(). +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nBehavior The behavior desired from the targeting session. Must be +/// a TARGET_BEHAVIOR_* constant. +void EnterTargetingModeByVarName(object oPC, string sVarName, int nBehavior = TARGET_BEHAVIOR_ADD); + +/// @brief Retrieve a targeting hook id. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @returns The targeting hook id assocaited with oPC's sVarName target list. +int GetTargetingHookID(object oPC, string sVarName); + +/// @brief Retrieve a targeting hook's sVarName. +/// @param nHookID The targeting hook's ID. +/// @returns The target list name sVarName associated with nHookID. +string GetTargetingHookVarName(int nHookID); + +/// @brief Retrieve a targeting hook's allowed object types. +/// @param nHookID The targeting hook's ID. +/// @returns A bitmap containing the allowed target types associated with +/// nHookID. +int GetTargetingHookObjectType(int nHookID); + +/// @brief Retrieve a targeting hook's remaining uses. +/// @param nHookID The targeting hook's ID. +/// @returns The number of uses remaining for targeting hook nHookID. +int GetTargetingHookUses(int nHookID); + +/// @brief Retrieve a targeting hook's script. +/// @param nHookID The targeting hook's ID. +/// @returns The script associated with targeting hook nHookID. +string GetTargetingHookScript(int nHookID); + +/// @brief Add a targeting hook to the global targeting hook list and save +/// targeting hook parameters for later use. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nObjectType A bitmasked value containing all object types allowed +/// to be targeted by this hook. +/// @param sScript The script that will be run when this target hook is +/// satisfied. +/// @param nUses The number of times this targeting hook is allowed to be used +/// before it is automatically deleted. Omitting this value will yield a +/// single use hook. Use -1 for an infinite-use hook. +/// @param nValidCursor A MOUSECURSOR_* cursor indicating a valid target. +/// @param nInvalidCursor A MOUSECURSOR_* cursor indicating an invalid target. +/// @returns A unique ID associated with the new targeting hook. +int AddTargetingHook(object oPC, string sVarName, int nObjectType = OBJECT_TYPE_ALL, string sScript = "", + int nUses = 1, int nValidCursor = MOUSECURSOR_MAGIC, int nInvalidCursor = MOUSECURSOR_NOMAGIC); + +/// @brief Save target data to the PC object as an object and location variable +/// defined by sVarName in AddTargetingHook(). Decrements remaining targeting +/// hook uses and, if required, deletes the targeting hook. +/// @param oPC The PC object associated with the target list. +/// @returns TRUE if OpC has a current targeting hook, FALSE otherwise. +int SatisfyTargetingHook(object oPC); + +/// @brief Retrieve a targeting list's object at index nIndex. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The index at which to retrieve the target object. +/// @returns The targeting's lists target at index nIndex, or the first +/// target on the list if nIndex is omitted. +object GetTargetingHookObject(object oPC, string sVarName, int nIndex = 1); + +/// @brief Retrieve a targeting list's location at index nIndex. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The index at which to retrieve the target location. +/// @returns The targeting's lists location at index nIndex, or the first +/// location on the list if nIndex is omitted. +location GetTargetingHookLocation(object oPC, string sVarName, int nIndex = 1); + +/// @brief Retrieve a targeting list's position at index nIndex. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The index at which to retrieve the target position. +/// @returns The targeting's lists position at index nIndex, or the first +/// position on the list if nIndex is omitted. +vector GetTargetingHookPosition(object oPC, string sVarName, int nIndex = 1); + +/// @brief Determine how many targets are on a target list. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @returns The number of targets associated with the saved as sVarName +/// on oPC. +// ---< CountTargetingHookTargets >--- +int CountTargetingHookTargets(object oPC, string sVarName); + +/// @brief Delete a targeting hook target. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The index at which to delete the target data. If omitted, +/// the first target on the list will be deleted. +/// @returns The number of targets remaining on oPC's sVarName target list +/// after deletion. +int DeleteTargetingHookTarget(object oPC, string sVarName, int nIndex = 1); + +/// @brief Retrieve the target list object's internal index. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param oObject The object to find on oPC's sVarName target list. +int GetTargetingHookIndex(object oPC, string sVarName, object oTarget); + +/// @brief Delete target list target data by internal index. +/// @param oPC The PC object associated with the target list. +/// @param sVarName The VarName associated with the target list. +/// @param nIndex The internal index of the target data to be deleted. This +/// index can be retrieved from GetTargetingHookIndex(). +/// @returns The number of targets remaining on oPC's sVarName target list +/// after deletion. +int DeleteTargetingHookTargetByIndex(object oPC, string sVarName, int nIndex); + +// ----------------------------------------------------------------------------- +// Private Function Definitions +// ----------------------------------------------------------------------------- + +sqlquery _PrepareTargetingQuery(string s) +{ + return SqlPrepareQueryObject(GetModule(), s); +} + +string _GetTargetingHookFieldData(int nHookID, string sField) +{ + string s = "SELECT " + sField + " " + + "FROM targeting_hooks " + + "WHERE nHookID = @nHookID;"; + sqlquery q = _PrepareTargetingQuery(s); + SqlBindInt(q, "@nHookID", nHookID); + + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +int _GetLastTargetingHookID() +{ + string s = "SELECT seq FROM sqlite_sequence WHERE name = @name;"; + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@name", "targeting_hooks"); + + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +string _GetTargetData(object oPC, string sVarName, string sField, int nIndex = 1) +{ + string s = "SELECT " + sField + " " + + "FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName " + + "LIMIT 1 OFFSET " + IntToString(nIndex) + ";"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +void _EnterTargetingMode(struct TargetingHook th, int nBehavior) +{ + SetLocalInt(th.oPC, TARGET_HOOK_ID, th.nHookID); + SetLocalInt(th.oPC, TARGET_HOOK_BEHAVIOR, nBehavior); + EnterTargetingMode(th.oPC, th.nObjectType, th.nValidCursor, th.nInvalidCursor); +} + +void _DeleteTargetingHookData(int nHookID) +{ + string s = "DELETE FROM targeting_hooks " + + "WHERE nHookID = @nHookID;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindInt(q, "@nHookID", nHookID); + SqlStep(q); +} + +void _ExitTargetingMode(int nHookID) +{ + struct TargetingHook th = GetTargetingHookDataByHookID(nHookID); + if (th.sScript != "") + { + Debug("Running post-targeting script " + th.sScript + " from Targeting Hook ID " + + IntToString(nHookID) + " on " + GetName(th.oPC) + " with varname " + th.sVarName); + RunTargetingHookScript(th.sScript, th.oPC); + } + else + Debug("No post-targeting script specified for Targeting Hook ID " + IntToString(nHookID) + " " + + "on " + GetName(th.oPC) + " with varname " + th.sVarName); + + DeleteTargetingHook(nHookID); + DeleteLocalInt(th.oPC, TARGET_HOOK_ID); + DeleteLocalInt(th.oPC, TARGET_HOOK_BEHAVIOR); +} + +// Reduces the number of targeting hooks remaining. When the remaining number is +// 0, the hook is automatically deleted. +int _DecrementTargetingHookUses(struct TargetingHook th, int nBehavior) +{ + int nUses = GetTargetingHookUses(th.nHookID); + + if (--nUses == 0) + { + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + Debug("Decrementing target hook uses for ID " + HexColorString(IntToString(th.nHookID), COLOR_CYAN) + + "\n Uses remaining -> " + (nUses ? HexColorString(IntToString(nUses), COLOR_CYAN) : HexColorString(IntToString(nUses), COLOR_RED_LIGHT)) + "\n"); + + _ExitTargetingMode(th.nHookID); + } + else + { + string s = "UPDATE targeting_hooks " + + "SET nUses = nUses - 1 " + + "WHERE nHookID = @nHookID;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindInt(q, "@nHookID", th.nHookID); + SqlStep(q); + + _EnterTargetingMode(th, nBehavior); + } + + return nUses; +} + +// ----------------------------------------------------------------------------- +// Public Function Definitions +// ----------------------------------------------------------------------------- + +// Temporary function for feedback purposes only +string ObjectTypeToString(int nObjectType) +{ + string sResult; + + if (nObjectType & OBJECT_TYPE_CREATURE) + sResult += (sResult == "" ? "" : ", ") + "Creatures"; + + if (nObjectType & OBJECT_TYPE_ITEM) + sResult += (sResult == "" ? "" : ", ") + "Items"; + + if (nObjectType & OBJECT_TYPE_TRIGGER) + sResult += (sResult == "" ? "" : ", ") + "Triggers"; + + if (nObjectType & OBJECT_TYPE_DOOR) + sResult += (sResult == "" ? "" : ", ") + "Doors"; + + if (nObjectType & OBJECT_TYPE_AREA_OF_EFFECT) + sResult += (sResult == "" ? "" : ", ") + "Areas of Effect"; + + if (nObjectType & OBJECT_TYPE_WAYPOINT) + sResult += (sResult == "" ? "" : ", ") + "Waypoints"; + + if (nObjectType & OBJECT_TYPE_PLACEABLE) + sResult += (sResult == "" ? "" : ", ") + "Placeables"; + + if (nObjectType & OBJECT_TYPE_STORE) + sResult += (sResult == "" ? "" : ", ") + "Stores"; + + if (nObjectType & OBJECT_TYPE_ENCOUNTER) + sResult += (sResult == "" ? "" : ", ") + "Encounters"; + + if (nObjectType & OBJECT_TYPE_TILE) + sResult += (sResult == "" ? "" : ", ") + "Tiles"; + + return sResult; +} + +void CreateTargetingDataTables(int bReset = FALSE) +{ + object oModule = GetModule(); + + if (bReset) + { + string sDropHooks = "DROP TABLE IF EXISTS targeting_hooks;"; + string sDropTargets = "DROP TABLE IF EXISTS targeting_targets;"; + + sqlquery q; + q = _PrepareTargetingQuery(sDropHooks); SqlStep(q); + q = _PrepareTargetingQuery(sDropTargets); SqlStep(q); + + DeleteLocalInt(oModule, "TARGETING_INITIALIZED"); + Warning(HexColorString("Targeting database tables have been dropped", COLOR_RED_LIGHT)); + } + + if (GetLocalInt(oModule, "TARGETING_INITIALIZED")) + return; + + string sData = "CREATE TABLE IF NOT EXISTS targeting_hooks (" + + "nHookID INTEGER PRIMARY KEY AUTOINCREMENT, " + + "sUUID TEXT, " + + "sVarName TEXT, " + + "nObjectType INTEGER, " + + "nUses INTEGER default '1', " + + "sScript TEXT, " + + "nValidCursor INTEGER, " + + "nInvalidCursor INTEGER, " + + "UNIQUE (sUUID, sVarName));"; + + string sTargets = "CREATE TABLE IF NOT EXISTS targeting_targets (" + + "nTargetID INTEGER PRIMARY KEY AUTOINCREMENT, " + + "sUUID TEXT, " + + "sVarName TEXT, " + + "sTargetObject TEXT, " + + "sTargetArea TEXT, " + + "vTargetLocation TEXT);"; + + sqlquery q; + q = _PrepareTargetingQuery(sData); SqlStep(q); + q = _PrepareTargetingQuery(sTargets); SqlStep(q); + + Debug(HexColorString("Targeting database tables have been created", COLOR_GREEN_LIGHT)); + SetLocalInt(oModule, "TARGETING_INITIALIZED", TRUE); +} + +struct TargetingHook GetTargetingHookDataByHookID(int nHookID) +{ + string s = "SELECT sUUID, sVarName, nObjectType, nUses, sScript, nValidCursor, nInvalidCursor " + + "FROM targeting_hooks " + + "WHERE nHookID = @nHookID;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindInt(q, "@nHookID", nHookID); + + struct TargetingHook th; + + if (SqlStep(q)) + { + th.nHookID = nHookID; + th.oPC = GetObjectByUUID(SqlGetString(q, 0)); + th.sVarName = SqlGetString(q, 1); + th.nObjectType = SqlGetInt(q, 2); + th.nUses = SqlGetInt(q, 3); + th.sScript = SqlGetString(q, 4); + th.nValidCursor = SqlGetInt(q, 5); + th.nInvalidCursor = SqlGetInt(q, 6); + } + else + Warning("Targeting data for target hook " + IntToString(nHookID) + " not found"); + + return th; +} + +struct TargetingHook GetTargetingHookDataByVarName(object oPC, string sVarName) +{ + int nHookID = GetTargetingHookID(oPC, sVarName); + return GetTargetingHookDataByHookID(nHookID); +} + +sqlquery GetTargetList(object oPC, string sVarName, int nIndex = -1) +{ + string s = "SELECT sTargetObject, sTargetArea, vTargetLocation " + + "FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName" + + (nIndex == -1 ? ";" : "LIMIT 1 OFFSET " + IntToString(nIndex)) + ";"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + + return q; +} + +int AddTargetToTargetList(object oPC, string sVarName, object oTarget, object oArea, vector vTarget) +{ + string s = "INSERT INTO targeting_targets (sUUID, sVarName, sTargetObject, sTargetArea, vTargetLocation) " + + "VALUES (@sUUID, @sVarName, @sTargetObject, @sTargetArea, @vTargetLocation);"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + SqlBindString(q, "@sTargetObject", ObjectToString(oTarget)); + SqlBindString(q, "@sTargetArea", ObjectToString(oArea)); + SqlBindVector(q, "@vTargetLocation", vTarget); + SqlStep(q); + + return CountTargetingHookTargets(oPC, sVarName); +} + +void DeleteTargetList(object oPC, string sVarName) +{ + string s = "DELETE FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + + SqlStep(q); +} + +void EnterTargetingModeByHookID(int nHookID, int nBehavior = TARGET_BEHAVIOR_ADD) +{ + struct TargetingHook th = GetTargetingHookDataByHookID(nHookID); + + if (th == TARGETING_HOOK_INVALID) + { + Warning("EnterTargetingModeByHookID::Unable to retrieve valid targeting data for " + + "targeting hook " + IntToString(nHookID)); + return; + } + + if (GetIsObjectValid(th.oPC)) + _EnterTargetingMode(th, nBehavior); +} + +void EnterTargetingModeByVarName(object oPC, string sVarName, int nBehavior = TARGET_BEHAVIOR_ADD) +{ + struct TargetingHook th = GetTargetingHookDataByVarName(oPC, sVarName); + + if (th == TARGETING_HOOK_INVALID) + { + Warning("EnterTargetingModeByVarName::Unable to retrieve valid targeting data for " + + "targeting hook " + sVarName + " on " + GetName(oPC)); + return; + } + + if (GetIsObjectValid(th.oPC)) + _EnterTargetingMode(th, nBehavior); +} + +int GetTargetingHookID(object oPC, string sVarName) +{ + string s = "SELECT nHookID " + + "FROM targeting_hooks " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +string GetTargetingHookVarName(int nHookID) +{ + return _GetTargetingHookFieldData(nHookID, "sVarName"); +} + +int GetTargetingHookObjectType(int nHookID) +{ + return StringToInt(_GetTargetingHookFieldData(nHookID, "nObjectType")); +} + +int GetTargetingHookUses(int nHookID) +{ + return StringToInt(_GetTargetingHookFieldData(nHookID, "nUses")); +} + +string GetTargetingHookScript(int nHookID) +{ + return _GetTargetingHookFieldData(nHookID, "sScript"); +} + +int AddTargetingHook(object oPC, string sVarName, int nObjectType = OBJECT_TYPE_ALL, string sScript = "", + int nUses = 1, int nValidCursor = MOUSECURSOR_MAGIC, int nInvalidCursor = MOUSECURSOR_NOMAGIC) +{ + CreateTargetingDataTables(); + + string s = "REPLACE INTO targeting_hooks (sUUID, sVarName, nObjectType, nUses, sScript, nValidCursor, nInvalidCursor) " + + "VALUES (@sUUID, @sVarName, @nObjectType, @nUses, @sScript, @nValidCursor, @nInvalidCursor);"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + SqlBindInt (q, "@nObjectType", nObjectType); + SqlBindInt (q, "@nUses", nUses); + SqlBindString(q, "@sScript", sScript); + SqlBindInt (q, "@nValidCursor", nValidCursor); + SqlBindInt (q, "@nInvalidCursor", nInvalidCursor); + SqlStep(q); + + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + { + Debug("Adding targeting hook ID " + HexColorString(IntToString(_GetLastTargetingHookID()), COLOR_CYAN) + + "\n sVarName -> " + HexColorString(sVarName, COLOR_CYAN) + + "\n nObjectType -> " + HexColorString(ObjectTypeToString(nObjectType), COLOR_CYAN) + + "\n sScript -> " + (sScript == "" ? HexColorString("[None]", COLOR_RED_LIGHT) : + HexColorString(sScript, COLOR_CYAN)) + + "\n nUses -> " + (nUses == -1 ? HexColorString("Unlimited", COLOR_CYAN) : + (nUses > 0 ? HexColorString(IntToString(nUses), COLOR_CYAN) : + HexColorString(IntToString(nUses), COLOR_RED_LIGHT))) + + "\n nValidCursor -> " + IntToString(nValidCursor) + + "\n nInvalidCursor -> " + IntToString(nInvalidCursor) + "\n"); + } + + return _GetLastTargetingHookID(); +} + +void DeleteTargetingHook(int nHookID) +{ + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + Debug("Deleting targeting hook ID " + HexColorString(IntToString(nHookID), COLOR_CYAN) + "\n"); + + _DeleteTargetingHookData(nHookID); +} + +int SatisfyTargetingHook(object oPC) +{ + int nHookID = GetLocalInt(oPC, TARGET_HOOK_ID); + if (nHookID == 0) + return FALSE; + + int nBehavior = GetLocalInt(oPC, TARGET_HOOK_BEHAVIOR); + + struct TargetingHook th = GetTargetingHookDataByHookID(nHookID); + + if (th == TARGETING_HOOK_INVALID) + { + Warning("SatisfyTargetingHook::Unable to retrieve valid targeting data for " + + "targeting hook " + IntToString(nHookID)); + return FALSE; + } + + string sVarName = th.sVarName; + object oTarget = GetTargetingModeSelectedObject(); + vector vTarget = GetTargetingModeSelectedPosition(); + + int bValid = TRUE; + + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + { + Debug("Targeted Object -> " + (GetIsObjectValid(oTarget) ? (GetIsPC(oTarget) ? HexColorString(GetName(oTarget), COLOR_GREEN_LIGHT) : HexColorString(GetTag(oTarget), COLOR_CYAN)) : HexColorString("OBJECT_INVALID", COLOR_RED_LIGHT)) + + "\n Type -> " + HexColorString(ObjectTypeToString(GetObjectType(oTarget)), COLOR_CYAN)); + Debug("Targeted Position -> " + (vTarget == Vector() ? HexColorString("POSITION_INVALID", COLOR_RED_LIGHT) : + HexColorString("(" + FloatToString(vTarget.x, 3, 1) + ", " + + FloatToString(vTarget.y, 3, 1) + ", " + + FloatToString(vTarget.z, 3, 1) + ")", COLOR_CYAN)) + "\n"); + } + + if (GetIsObjectValid(oTarget)) + { + if (nBehavior == TARGET_BEHAVIOR_ADD) + { + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + { + object oArea = GetArea(oTarget); + + Debug(HexColorString("Saving targeted object and position to list [" + th.sVarName + "]:", COLOR_CYAN) + + "\n Tag -> " + HexColorString(GetTag(oTarget), COLOR_CYAN) + + "\n Location -> " + HexColorString(JsonDump(LocationToJson(Location(oArea, vTarget, 0.0))), COLOR_CYAN) + + "\n Area -> " + HexColorString((GetIsObjectValid(oArea) ? GetTag(oArea) : "AREA_INVALID"), COLOR_CYAN) + "\n"); + } + + AddTargetToTargetList(oPC, sVarName, oTarget, GetArea(oPC), vTarget); + } + else if (nBehavior == TARGET_BEHAVIOR_DELETE) + { + if (GetArea(oTarget) == oTarget) + Warning("Location/Tile targets cannot be deleted; select a game object"); + else + { + Debug(HexColorString("Attempting to delete targeted object and position from list [" + th.sVarName + "]:", COLOR_CYAN)); + int nIndex = GetTargetingHookIndex(oPC, sVarName, oTarget); + if (nIndex == 0 && IsDebugging(DEBUG_LEVEL_DEBUG)) + Debug(" > " + HexColorString("Target " + (GetIsPC(oTarget) ? GetName(oTarget) : GetTag(oTarget)) + " not found " + + "on list [" + th.sVarName + "]; removal aborted", COLOR_RED_LIGHT)); + else + { + DeleteTargetingHookTargetByIndex(oPC, sVarName, nIndex); + + if (IsDebugging(DEBUG_LEVEL_DEBUG)) + Debug(" > " + HexColorString("Target " + (GetIsPC(oTarget) ? GetName(oTarget) : GetTag(oTarget)) + " removed from " + + "list [" + th.sVarName + "]", COLOR_GREEN_LIGHT)); + } + } + } + } + else + bValid = FALSE; + + if (!bValid) + _ExitTargetingMode(nHookID); + else + { + if (th.nUses == -1) + _EnterTargetingMode(th, nBehavior); + else + _DecrementTargetingHookUses(th, nBehavior); + } + + return TRUE; +} + +int DeleteTargetingHookTargetByIndex(object oPC, string sVarName, int nIndex) +{ + string s = "DELETE FROM targeting_targets " + + "WHERE nTargetID = @nTargetID;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindInt(q, "@nTargetID", nIndex); + SqlStep(q); + + return CountTargetingHookTargets(oPC, sVarName); +} + +int GetTargetingHookIndex(object oPC, string sVarName, object oTarget) +{ + string s = "SELECT nTargetID " + + "FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName " + + "AND sTargetObject = @sTargetObject;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + SqlBindString(q, "@sTargetObject", ObjectToString(oTarget)); + + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +object GetTargetingHookObject(object oPC, string sVarName, int nIndex = 1) +{ + return StringToObject(_GetTargetData(oPC, sVarName, "sTargetObject", nIndex)); +} + +location GetTargetingHookLocation(object oPC, string sVarName, int nIndex = 1) +{ + sqlquery q = GetTargetList(oPC, sVarName, 1); + if (SqlStep(q)) + { + object oArea = StringToObject(SqlGetString(q, 1)); + vector vTarget = SqlGetVector(q, 2); + + return Location(oArea, vTarget, 0.0); + } + + return Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector GetTargetingHookPosition(object oPC, string sVarName, int nIndex = 1) +{ + sqlquery q = GetTargetList(oPC, sVarName, 1); + if (SqlStep(q)) + return SqlGetVector(q, 2); + + return Vector(); +} + +int CountTargetingHookTargets(object oPC, string sVarName) +{ + string s = "SELECT COUNT (nTargetID) " + + "FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName;"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +int DeleteTargetingHookTarget(object oPC, string sVarName, int nIndex = 1) +{ + string s = "DELETE FROM targeting_targets " + + "WHERE sUUID = @sUUID " + + "AND sVarName = @sVarName " + + "LIMIT 1 OFFSET " + IntToString(nIndex) + ";"; + + sqlquery q = _PrepareTargetingQuery(s); + SqlBindString(q, "@sUUID", GetObjectUUID(oPC)); + SqlBindString(q, "@sVarName", sVarName); + SqlStep(q); + + return CountTargetingHookTargets(oPC, sVarName); +} diff --git a/_module/nss/util_i_timers.nss b/_module/nss/util_i_timers.nss new file mode 100644 index 0000000..bc2a965 --- /dev/null +++ b/_module/nss/util_i_timers.nss @@ -0,0 +1,475 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_timers.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @author Ed Burke (tinygiant98) +/// @brief Functions for running scripts on an interval. +/// ---------------------------------------------------------------------------- +/// @details +/// ## Concept +/// Timers are a way of running a script repeatedly on an interval. A timer can +/// be created on an object. Once started, it will continue to run until it is +/// finished iterating or until killed manually. Each time the timer elapses, +/// its action will run. By default, this action is to simply run a script. +/// +/// ## Basic Usage +/// +/// ### Creating a Timer +/// You can create a timer using `CreateTimer()`. This function takes the object +/// that should run the timer, the script that should execute when the timer +/// elapses, the interval between ticks, and the total number of iterations. It +/// returns the ID for the timer, which is used to reference it in the database. +/// You should save this timer for later use. +/// +/// ```nwscript +/// // The following creates a timer on oPC that will run the script "foo" every +/// // 6 seconds for 4 iterations. +/// int nTimerID = CreateTimer(oPC, "foo", 6.0, 4); +/// ``` +/// +/// A timer created with 0 iterations will run until stopped or killed. +/// +/// ## Starting a Timer +/// Timers will not run until they are started wiuth `StartTimer()`. This +/// function takes the ID of the timer returned from `CreateTimer()`. If the +/// second parameter, `bInstant`, is TRUE, the timer will elapse immediately; +/// otherwise, it will elapse when its interval is complete: +/// +/// ```nwscript +/// StartTimer(nTimerID); +/// ``` +/// +/// ### Stopping a Timer +/// Stopping a timer with `StopTimer()` will suspend its execution: +/// ```nwscript +/// StopTimer(nTimerID); +/// ``` +/// You can restart the timer later using `StartTimer()` to resume any remaining +/// iterations. If you want to start again from the beginning, you can call +/// `ResetTimer()` first: +/// ```nwscript +/// ResetTimer(nTimerID); +/// StartTimer(nTimerID); +/// ``` +/// +/// ### Destroying a Timer +/// Calling `KillTimer()` will clean up all data associated with the timer. A +/// timer cannot be restarted after it is killed; you will have to create and +/// start a new one. +/// ```nwscript +/// KillTimer(nTimerID); +/// ``` +/// +/// Timers automatically kill themselves when they are finished iterating or +/// when the object they are executed on is no longer valid. You only need to +/// use `KillTimer()` if you want to destroy it before it is done iterating or +/// if the timer is infinite. +/// +/// ## Advanced Usage +/// By default, timer actions are handled by passing them to `ExecuteScript()`. +/// However, the final parameter of the `CreateTimer()` function allows you to +/// specify a handler script. If this parameter is not blank, the handler will +/// be called using `ExecuteScript()` and the action will be available to it as +/// a script parameter. +/// +/// For example, the Core Framework allows timers to run event hooks by calling +/// the handler script `core_e_timerhook`, which is as follows: +/// ```nwscript +/// #include "core_i_framework" +/// +/// void main() +/// { +/// string sEvent = GetScriptParam(TIMER_ACTION); +/// string sSource = GetScriptParam(TIMER_SOURCE); +/// object oSource = StringToObject(sSource); +/// RunEvent(sEvent, oSource); +/// } +/// ``` +/// +/// To make this easier, `core_i_framework` contains an alias to `CreateTimer()` +/// called `CreateEventTimer()` that sets the handler script. You can create +/// your own aliases in the same way. + +#include "util_i_sqlite" +#include "util_i_debug" +#include "util_i_datapoint" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const string TIMER_DATAPOINT = "*Timers"; +const string TIMER_INIT = "*TimersInitialized"; +const string TIMER_LAST = "*TimerID"; +const string TIMER_ACTION = "*TimerAction"; +const string TIMER_SOURCE = "*TimerSource"; + +// ----------------------------------------------------------------------------- +// Global Variables +// ----------------------------------------------------------------------------- + +// Running timers are AssignCommand()'ed to this datapoint. This ensures that +// even if the object that issued the StartTimer() becomes invalid, the timer +// will continue to run. +object TIMERS = GetDatapoint(TIMER_DATAPOINT, GetModule(), FALSE); + +// ----------------------------------------------------------------------------- +// Public Function Declarations +// ----------------------------------------------------------------------------- + +/// @brief Create a table for timers in the module's volatile database. +/// @param bReset If TRUE, will drop the existing timers table. +/// @note This function will be run automatically the first timer one of the +/// functions in this file is called. You only need to call this if you need +/// the table created earlier (e.g., because another table references it). +void CreateTimersTable(int bReset = FALSE); + +/// @brief Create a timer that fires on a target at regular intervals. +/// @details After a timer is created, you will need to start it to get it to +/// run. You cannot create a timer on an invalid target or with a +/// non-positive interval value. +/// @param oTarget The object the action will run on. +/// @param sAction The action to execute when the timer elapses. +/// @param fInterval The number of seconds between iterations. +/// @param nIterations the number of times the timer can elapse. 0 means no +/// limit. If nIterations is 0, fInterval must be greater than or equal to +/// 6.0. +/// @param fJitter A random number of seconds between 0.0 and fJitter to add to +/// fInterval between executions. Leave at 0.0 for no jitter. +/// @param sHandler A handler script to execute sAction. If "", sAction will be +/// called using ExecuteScript() instead. +/// @returns the ID of the timer. Save this so it can be used to start, stop, or +/// kill the timer later. +int CreateTimer(object oTarget, string sAction, float fInterval, int nIterations = 0, float fJitter = 0.0, string sHandler = ""); + +/// @brief Return if a timer exists. +/// @param nTimerID The ID of the timer in the database. +int GetIsTimerValid(int nTimerID); + +/// @brief Start a timer, executing its action each interval until finished +/// iterating, stopped, or killed. +/// @param nTimerID The ID of the timer in the database. +/// @param bInstant If TRUE, execute the timer's action immediately. +void StartTimer(int nTimerID, int bInstant = TRUE); + +/// @brief Suspend execution of a timer. +/// @param nTimerID The ID of the timer in the database. +/// @note This does not destroy the timer, only stops it from iterating or +/// executing its action. +void StopTimer(int nTimerID); + +/// @brief Reset the number or remaining iterations on a timer. +/// @param nTimerID The ID of the timer in the database. +void ResetTimer(int nTimerID); + +/// @brief Delete a timer. +/// @details This results in all information about the given timer being +/// deleted. Since the information is gone, the action associated with that +/// timer ID will not get executed again. +/// @param nTimerID The ID of the timer in the database. +void KillTimer(int nTimerID); + +/// @brief Return whether a timer will run infinitely. +/// @param nTimerID The ID of the timer in the database. +int GetIsTimerInfinite(int nTimerID); + +/// @brief Return the remaining number of iterations for a timer. +/// @details If called during a timer script, will not include the current +/// iteration. Returns -1 if nTimerID is not a valid timer ID. Returns 0 if +/// the timer is set to run indefinitely, so be sure to check for this with +/// GetIsTimerInfinite(). +/// @param nTimerID The ID of the timer in the database. +int GetTimerRemaining(int nTimerID); + +/// @brief Sets the remaining number of iterations for a timer. +/// @param nTimerID The ID of the timer in the database. +/// @param nRemaining The remaining number of iterations. +void SetTimerRemaining(int nTimerID, int nRemaining); + +// ----------------------------------------------------------------------------- +// Private Function Implementations +// ----------------------------------------------------------------------------- + +// Private function used by StartTimer(). +void _TimerElapsed(int nTimerID, int nRunID, int bFirstRun = FALSE) +{ + // Timers are fired on a delay, so it's possible that the timer was stopped + // and restarted before the delayed call could fail due to the timer being + // stopped. We increment the run_id whenever the timer is started and pass + // it along to the delayed calls so they can check if they are still valid. + sqlquery q = SqlPrepareQueryModule("SELECT * FROM timers " + + "WHERE timer_id = @timer_id AND run_id = @run_id AND running = 1;"); + SqlBindInt(q, "@timer_id", nTimerID); + SqlBindInt(q, "@run_id", nRunID); + + // The timer was killed or stopped + if (!SqlStep(q)) + return; + + string sTimerID = IntToString(nTimerID); + string sAction = SqlGetString(q, 3); + string sHandler = SqlGetString(q, 4); + string sTarget = SqlGetString(q, 5); + string sSource = SqlGetString(q, 6); + float fInterval = SqlGetFloat (q, 7); + float fJitter = SqlGetFloat (q, 8); + int nIterations = SqlGetInt (q, 9); + int nRemaining = SqlGetInt (q, 10); + int bIsPC = SqlGetInt (q, 11); + object oTarget = StringToObject(sTarget); + object oSource = StringToObject(sSource); + + string sMsg = + "\n Target: " + sTarget + + " (" + (GetIsObjectValid(oTarget) ? GetName(oTarget) : "INVALID") + ")" + + "\n Source: " + sSource + + " (" + (GetIsObjectValid(oTarget) ? GetName(oSource) : "INVALID") + ")" + + "\n Action: " + sAction + + "\n Handler: " + sHandler; + + if (!GetIsObjectValid(oTarget) || (bIsPC && !GetIsPC(oTarget))) + { + Warning("Target for timer " + sTimerID + " no longer valid:" + sMsg); + KillTimer(nTimerID); + return; + } + + // If we're running infinitely or we have more runs remaining... + if (!nIterations || nRemaining) + { + string sIterations = (nIterations ? IntToString(nIterations) : "Infinite"); + if (!bFirstRun) + { + Notice("Timer " + sTimerID + " elapsed" + sMsg + + "\n Iteration: " + + (nIterations ? IntToString(nIterations - nRemaining + 1) : "INFINITE") + + "/" + sIterations); + + // If we're not running an infinite number of times, decrement the + // number of iterations we have remaining + if (nIterations) + SetTimerRemaining(nTimerID, nRemaining - 1); + + // Run the timer handler + SetScriptParam(TIMER_LAST, IntToString(nTimerID)); + SetScriptParam(TIMER_ACTION, sAction); + SetScriptParam(TIMER_SOURCE, sSource); + ExecuteScript(sHandler != "" ? sHandler : sAction, oTarget); + + // In case one of those scripts we just called reset the timer... + if (nIterations) + nRemaining = GetTimerRemaining(nTimerID); + } + + // If we have runs left, call our timer's next iteration. + if (!nIterations || nRemaining) + { + // Account for any jitter + fJitter = IntToFloat(Random(FloatToInt(fJitter * 10) + 1)) / 10.0; + fInterval += fJitter; + + Notice("Scheduling next iteration for timer " + sTimerID + ":" + sMsg + + "\n Delay: " + FloatToString(fInterval, 0, 1) + + "\n Remaining: " + + (nIterations ? (IntToString(nRemaining)) : "INFINITE") + + "/" + sIterations); + + DelayCommand(fInterval, _TimerElapsed(nTimerID, nRunID)); + return; + } + } + + // We have no more runs left! Kill the timer to clean up. + Debug("Timer " + sTimerID + " expired:" + sMsg); + KillTimer(nTimerID); +} + +// ----------------------------------------------------------------------------- +// Public Function Implementations +// ----------------------------------------------------------------------------- + +void CreateTimersTable(int bReset = FALSE) +{ + if (GetLocalInt(TIMERS, TIMER_INIT) && !bReset) + return; + + // StartTimer() assigns the timer tick to TIMERS, so by deleting it, we are + // able to cancel all currently running timers. + DestroyObject(TIMERS); + + SqlCreateTableModule("timers", + "timer_id INTEGER PRIMARY KEY AUTOINCREMENT, " + + "run_id INTEGER NOT NULL DEFAULT 0, " + + "running BOOLEAN NOT NULL DEFAULT 0, " + + "action TEXT NOT NULL, " + + "handler TEXT NOT NULL, " + + "target TEXT NOT NULL, " + + "source TEXT NOT NULL, " + + "interval REAL NOT NULL, " + + "jitter REAL NOT NULL, " + + "iterations INTEGER NOT NULL, " + + "remaining INTEGER NOT NULL, " + + "is_pc BOOLEAN NOT NULL DEFAULT 0", bReset); + + TIMERS = CreateDatapoint(TIMER_DATAPOINT); + SetDebugPrefix(HexColorString("[Timers]", COLOR_CYAN), TIMERS); + SetLocalInt(TIMERS, TIMER_INIT, TRUE); +} + +int CreateTimer(object oTarget, string sAction, float fInterval, int nIterations = 0, float fJitter = 0.0, string sHandler = "") +{ + string sSource = ObjectToString(OBJECT_SELF); + string sTarget = ObjectToString(oTarget); + string sDebug = + "\n OBJECT_SELF: " + sSource + " (" + GetName(OBJECT_SELF) + ")" + + "\n oTarget: " + sTarget + + " (" + (GetIsObjectValid(oTarget) ? GetName(oTarget) : "INVALID") + ")" + + "\n sAction: " + sAction + + "\n sHandler: " + sHandler + + "\n nIterations: " + (nIterations ? IntToString(nIterations) : "Infinite") + + "\n fInterval: " + FloatToString(fInterval, 0, 1) + + "\n fJitter: " + FloatToString(fJitter, 0, 1); + + // Sanity checks: don't create the timer if... + // 1. the target is invalid + // 2. the interval is not greater than 0.0 + // 3. the number of iterations is non-positive + // 4. the interval is more than once per round and the timer is infinite + string sError; + if (!GetIsObjectValid(oTarget)) + sError = "oTarget is invalid"; + else if (fInterval <= 0.0) + sError = "fInterval must be positive"; + else if (fInterval + fJitter <= 0.0) + sError = "fJitter is too low for fInterval"; + else if (nIterations < 0) + sError = "nIterations is negative"; + else if (fInterval < 6.0 && !nIterations) + sError = "fInterval is too short for infinite executions"; + + if (sError != "") + { + CriticalError("CreateTimer() failed:\n Error: " + sError + sDebug); + return 0; + } + + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule("INSERT INTO timers " + + "(action, handler, target, source, interval, jitter, iterations, remaining, is_pc) " + + "VALUES (@action, @handler, @target, @source, @interval, @jitter, @iterations, @remaining, @is_pc) " + + "RETURNING timer_id;"); + SqlBindString(q, "@action", sAction); + SqlBindString(q, "@handler", sHandler); + SqlBindString(q, "@target", sTarget); + SqlBindString(q, "@source", sSource); + SqlBindFloat (q, "@interval", fInterval); + SqlBindFloat (q, "@jitter", fJitter); + SqlBindInt (q, "@iterations", nIterations); + SqlBindInt (q, "@remaining", nIterations); + SqlBindInt (q, "@is_pc", GetIsPC(oTarget)); + + int nTimerID = SqlStep(q) ? SqlGetInt(q, 0) : 0; + if (nTimerID > 0) + Notice("Created timer " + IntToString(nTimerID) + sDebug); + + return nTimerID; +} + +int GetIsTimerValid(int nTimerID) +{ + // Timer IDs less than or equal to 0 are always invalid. + if (nTimerID <= 0) + return FALSE; + + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "SELECT 1 FROM timers WHERE timer_id = @timer_id;"); + SqlBindInt(q, "@timer_id", nTimerID); + return SqlStep(q) ? SqlGetInt(q, 0) : FALSE; +} + +void StartTimer(int nTimerID, int bInstant = TRUE) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "UPDATE timers SET running = 1, run_id = run_id + 1 " + + "WHERE timer_id = @timer_id AND running = 0 RETURNING run_id;"); + SqlBindInt(q, "@timer_id", nTimerID); + + if (SqlStep(q)) + { + Notice("Started timer " + IntToString(nTimerID)); + AssignCommand(TIMERS, _TimerElapsed(nTimerID, SqlGetInt(q, 0), !bInstant)); + } + else + { + string sDebug = "StartTimer(" + IntToString(nTimerID) + ")"; + if (GetIsTimerValid(nTimerID)) + Error(sDebug + "failed: timer is already running"); + else + Error(sDebug + " failed: timer id does not exist"); + } +} + +void StopTimer(int nTimerID) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "UPDATE timers SET running = 0 " + + "WHERE timer_id = @timer_id RETURNING 1;"); + SqlBindInt(q, "@timer_id", nTimerID); + if (SqlStep(q)) + Notice("Stopping timer " + IntToString(nTimerID)); +} + +void ResetTimer(int nTimerID) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "UPDATE timers SET remaining = timers.iterations " + + "WHERE timer_id = @timer_id AND iterations > 0 RETURNING remaining;"); + SqlBindInt(q, "@timer_id", nTimerID); + if (SqlStep(q)) + { + Notice("ResetTimer(" + IntToString(nTimerID) + ") successful: " + + IntToString(SqlGetInt(q, 0)) + " iterations remaining"); + } +} + +void KillTimer(int nTimerID) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "DELETE FROM timers WHERE timer_id = @timer_id RETURNING 1;"); + SqlBindInt(q, "@timer_id", nTimerID); + if (SqlStep(q)) + Notice("Killing timer " + IntToString(nTimerID)); +} + +int GetIsTimerInfinite(int nTimerID) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "SELECT iterations FROM timers WHERE timer_id = @timer_id;"); + SqlBindInt(q, "@timer_id", nTimerID); + return SqlStep(q) ? !SqlGetInt(q, 0) : FALSE; +} + +int GetTimerRemaining(int nTimerID) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "SELECT remaining FROM timers WHERE timer_id = @timer_id;"); + SqlBindInt(q, "@timer_id", nTimerID); + return SqlStep(q) ? SqlGetInt(q, 0) : -1; +} + +void SetTimerRemaining(int nTimerID, int nRemaining) +{ + CreateTimersTable(); + sqlquery q = SqlPrepareQueryModule( + "UPDATE timers SET remaining = @remaining " + + "WHERE timer_id = @timer_id AND iterations > 0;"); + SqlBindInt(q, "@timer_id", nTimerID); + SqlBindInt(q, "@remaining", nRemaining); + SqlStep(q); +} diff --git a/_module/nss/util_i_times.nss b/_module/nss/util_i_times.nss new file mode 100644 index 0000000..859ba07 --- /dev/null +++ b/_module/nss/util_i_times.nss @@ -0,0 +1,1139 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_times.nss +/// @author Michael A. Sinclair (Squatting Monk) +/// @brief Functions for managing times, dates, and durations. +/// ---------------------------------------------------------------------------- +/// @details +/// # Concepts +/// - Time: a struct value that may represent either a *calendar time* or a +/// *duration*. A Time has a field for years, months, day, hours, minutes, +/// seconds, and milliseconds. A Time also has a field to set the minutes per +/// hour (defaults to the module setting, which defaults to 2). +/// - Calendar Time: A Time representing a particular *moment in time* as +/// measured using the game calendar and clock. In a calendar Time, the month +/// and day count from 1, while all other units count from 0 (including +/// years, since NWN allows year 0). A calendar Time must always be positive. +/// - Duration Time: A Time representing an *amount of time*. All units in a +/// duration Time count from 0. A duration Time may be negative, representing +/// going back in time. This can be useful for calculations. A duration Time +/// can be converted to seconds to pass it to game functions that expect a +/// time, such as `DelayCommand()`, `PlayAnimation()`, etc. +/// - Game Time: a Time (either calendar Time or duration Time) with a minutes +/// per hour setting of 60. This allows you to convert the time shown by the +/// game clock into a time that matches how the characters in the game would +/// perceive it. For example, with the default minutes per hour setting of 2, +/// the Time "13:01" would correspond to a Game Time of "13:30". +/// - Normalizing Times: You can normalize a Time to ensure none of its units +/// are overflowing their bounds. For example, a Time with a Minute field of 0 +/// and Second field of 90 would be normalized to a Minute of 1 and a Second +/// of 30. Normalizing a Time also causes all non-zero units to take the same +/// sign (either positive or negative), so a Time with a Minute of 1 and a +/// Second of -30 would normalize to a Second of 30. When normalizing a Time, +/// you can also change the minutes per hour setting. This is how the +/// functions in this file convert between Time and Game Time. +/// +/// **Note**: For brevity, some functions have a `Time` variant and a +/// `Duration` variant. In these cases, the `Time` variant refers to a +/// calendar Time (e.g., `StringToTime()` converts to a calendar Time while +/// `StringToDuration()` refers to a duration Time). If no `Duration` variant +/// of the function is present, the function may refer to a calendar Time *or* a +/// duration Time (e.g., `TimeToString()` accepts both types). +/// ---------------------------------------------------------------------------- +/// # Usage +/// +/// ## Creating a Time +/// You can create a calendar Time using `GetTime()` and a duration Time with +/// `GetDuration()`: +/// ```nwscript +/// struct Time t = GetTime(1372, 6, 1, 13); +/// struct Time d = GetDuration(1372, 5, 0, 13); +/// ``` +/// +/// You could also parse an ISO 8601 time string into a calendar Time or +/// duration Time: +/// ```nwscript +/// struct Time tTime = StringToTime("1372-06-01 13:00:00:000"); +/// struct Time tDur = StringToDuration("1372-05-00 13:00:00:000"); +/// +/// // Negative durations are allowed: +/// struct Time tNeg = StringToDuration("-1372-05-00 13:00:00:000"); +/// +/// // Missing units are assumed to be their lowest bound: +/// struct Time a = StringToTime("1372-06-01 00:00:00:000"); +/// struct Time b = StringToTime("1372-06-01"); +/// struct Time c = StringToTime("1372-06"); +/// Assert(a == b); +/// Assert(b == c); +/// ``` +/// +/// You can also create a Time manually by declaring a new Time struct and +/// setting the fields independently: +/// ```nwscript +/// struct Time t; +/// t.Type = TIME_TYPE_CALENDAR; +/// t.Year = 1372; +/// t.Month = 6; +/// t.Day = 1; +/// t.Hour = 13; +/// // ... +/// ``` +/// +/// When not using the `GetTime()` function, it's a good idea to normalize the +/// resultant Time to distribute the field values correctly: +/// ```nwscript +/// struct Time t = NewTime(); +/// t.Second = 90; +/// +/// t = NormalizeTime(t); +/// Assert(t.Minute == 1); +/// Assert(t.Second == 30); +/// ``` +/// +/// ## Converting Between Time and Game Time +/// +/// ```nwscript +/// // Assuming the default module setting of 2 minutes per hour +/// struct Time tTime = StringToTime("1372-06-01 13:01:00:000"); +/// Assert(tTime.Hour == 13); +/// Assert(tTime.Minute == 1); +/// +/// struct Time tGame = TimeToGameTime(tTime); +/// Assert(tGame.Hour == 13); +/// Assert(tGame.Minute == 30); +/// +/// struct tBack = GameTimeToTime(tGame); +/// Assert(tTime == tBack); +/// ``` +/// +/// ## Getting the Current Time +/// ```nwscript +/// struct Time tTime = GetCurrentTime(); +/// struct Time tGame = GetCurrentGameTime(); +/// ``` +/// +/// ## Setting the Current Time +/// @note You can only set the time forward in NWN. +/// +/// ```nwscript +/// struct Time t = StringToTime("2022-08-25 13:00:00:000"); +/// SetCurrentTime(t); +/// ``` +/// +/// Alternatively, you can advance the current Time by a duration Time: +/// ```nwscript +/// AdvanceCurrentTime(FloatToDuration(120.0)); +/// ``` +/// +/// ## Dropping units from a Time +/// You can reduce the precision of a Time. Units smaller than the precision +/// limit will be at their lower bound: +/// ```nwscript +/// struct Time a = GetTime(1372, 6, 1, 13); +/// struct Time b = GetTime(1372, 6, 1); +/// struct Time c = GetPrecisionTime(a, TIME_UNIT_DAY); +/// struct Time d = GetPrecisionTime(a, TIME_UNIT_MONTH); +/// Assert(a != b); +/// Assert(b == c); +/// Assert(b == d); +/// ``` +/// +/// ## Saving a Time +/// The easiest way to save a Time and get it later is to use the +/// `SetLocalTime()` and `GetLocalTime()` functions. These functions convert a +/// Time into json and save it as a local variable. +/// +/// In this example, we save the server start time OnModuleLoad and then get it +/// later: +/// ```nwscript +/// // OnModuleLoad +/// SetLocalTime(GetModule(), "ServerStart", GetCurrentTime()); +/// +/// // later on... +/// struct Time tServerStart = GetLocalTime(GetModule(), "ServerStart"); +/// ``` +/// +/// If you want to store a Time in a database, you can convert it into json or +/// into a string before passing it to a query. The json method is preferable +/// for persistent storage, since it is guaranteed to be correct if the module's +/// minutes per hour setting changes after the value is stored: +/// ```nwscript +/// struct Time tTime = GetCurrentTime(); +/// json jTime = TimeToJson(tTime); +/// string sSql = "INSERT INTO data (varname, value) VALUES ('ServerTime', @time);"; +/// sqlquery q = SqlPrepareQueryCampaign("mydb", sSql); +/// SqlBindJson(q, "@time", jTime); +/// SqlStep(q); +/// ``` +/// +/// You can then convert the json back into a Time: +/// ```nwscript +/// string Time tTime; +/// string sSql = "SELECT value FROM data WHERE varname='ServerTime';"; +/// sqlquery q = SqlPrepareQueryCampaign("mydb", sSql); +/// if (SqlStep(q)) +/// tTime = JsonToTime(SqlGetJson(q, 0)); +/// ``` +/// +/// For simpler applications (such as saving to the module's volatile database), +/// converting to a string works fine and could even be preferable since you can +/// use sqlite's `<`, `>`, and `=` operators to check if one time is before, +/// after, or equal to another. +/// ```nwscript +/// struct Time tTime = GetCurrentTime(); +/// string sTime = TimeToString(); +/// string sSql = "INSERT INTO data (varname, value) VALUES ('ServerTime', @time);"; +/// sqlquery q = SqlPrepareQueryCampaign("mydb", sSql); +/// SqlBindString(q, "@time", sTime); +/// SqlStep(q); +/// ``` +/// +/// ## Comparing Times +/// To check if one time is before or after another: +/// ```nwscript +/// struct Time a = StringToTime("1372-06-01 13:00:00:000"); +/// struct Time b = StringToTime("1372-06-01 13:01:30:500"); +/// Assert(GetIsTimeBefore(a, b)); +/// Assert(!GetIsTimeAfter(a, b)); +/// ``` +/// +/// To check if two times are equal: +/// ```nwscript +/// struct Time a = StringToTime("1372-06-01 13:00:00:000"); +/// struct Time b = StringToTime("1372-06-01 13:01:00:000"); +/// struct Time c = TimeToGameTime(b); +/// +/// Assert(!GetIsTimeEqual(a, b)); +/// Assert(GetIsTimeEqual(b, c)); +/// +/// // To check for exactly equal: +/// Assert(b != c); +/// ``` +/// +/// To check the amount of time between two Times: +/// ```nwscript +/// struct Time a = StringToTime("1372-06-01 13:00:00:000"); +/// struct Time b = StringToTime("1372-06-01 13:01:30:500"); +/// struct Time tDur = GetDurationBetween(a, b); +/// Assert(DurationToFloat(tDur) == 90.5); +/// ``` +/// +/// To check if a duration has passed since a Time: +/// ```nwscript +/// int CheckForMinRestTime(object oPC, float fMinTime) +/// { +/// struct Time tSince = GetDurationSince(GetLocalTime(oPC, "LastRest")); +/// return DurationToFloat(tSince) >= fMinTime; +/// } +/// ``` +/// +/// To calculate the duration until a Time is reached: +/// ```nwscript +/// struct Time tMidnight = GetTime(GetCalendarYear(), GetCalendarMonth(), GetCalendarDay() + 1); +/// struct Time tDurToMidnight = GetDurationUntil(tMidnight); +/// float fDurToMidnight = DurationToFloat(tDurToMidnight); +/// ``` +/// ---------------------------------------------------------------------------- + +#include "util_i_strings" +#include "util_i_debug" + +// ----------------------------------------------------------------------------- +// Types +// ----------------------------------------------------------------------------- + +/// @struct Time +/// @brief A datatype representing either an amount of time or a moment in time. +/// @note Times with a Type field of TIME_TYPE_DURATION represent an amount of +/// time as represented on a stopwatch. All duration units count from 0. +/// @note Times with a Type field of TIME_TYPE_CALENDAR represent a moment in +/// time as represented on a calendar. This means the month and day count +/// from 1, but all other units count from 0 (including the year, since NWN +/// allows year 0). +struct Time +{ + int Type; ///< TIME_TYPE_DURATION || TIME_TYPE_CALENDAR + int Year; ///< 0..32000 + int Month; ///< 0..11 for duration Times, 1..12 for calendar Times + int Day; ///< 0..27 for duration Times, 1..28 for calendar Times + int Hour; ///< 0..23 + int Minute; ///< 0..MinsPerHour + int Second; ///< 0..59 + int Millisecond; ///< 0..999 + int MinsPerHour; ///< The minutes per hour setting: 1..60 +}; + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// These are the units in a valid Time. +const int TIME_UNIT_YEAR = 0; +const int TIME_UNIT_MONTH = 1; +const int TIME_UNIT_DAY = 2; +const int TIME_UNIT_HOUR = 3; +const int TIME_UNIT_MINUTE = 4; +const int TIME_UNIT_SECOND = 5; +const int TIME_UNIT_MILLISECOND = 6; + +// These are the types of Times. +const int TIME_TYPE_DURATION = 0; ///< Represents an amount of time +const int TIME_TYPE_CALENDAR = 1; ///< Represents a moment in time + +// Prefix for local variables to avoid collision +const string TIME_PREFIX = "*Time: "; + +// These are field names for json objects +const string TIME_TYPE = "Type"; +const string TIME_YEAR = "Year"; +const string TIME_MONTH = "Month"; +const string TIME_DAY = "Day"; +const string TIME_HOUR = "Hour"; +const string TIME_MINUTE = "Minute"; +const string TIME_SECOND = "Second"; +const string TIME_MILLISECOND = "Millisecond"; +const string TIME_MINSPERHOUR = "MinsPerHour"; + +/// Uninitialized Time value. Can be compared to Times returned from functions +/// to see if the Time is valid. +struct Time TIME_INVALID; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Convert hours to minutes. +/// @param nHours The number of hours to convert. +/// @note The return value varies depending on the module's time settings. +int HoursToMinutes(int nHours = 1); + +// ----- Times ----------------------------------------------------------------- + +/// @brief Return whether any unit in a Time is less than its lower bound. +/// @param t The Time to check. +int GetAnyTimeUnitNegative(struct Time t); + +/// @brief Return whether any unit in a Time is greater than its lower bound. +/// @param t The Time to check. +int GetAnyTimeUnitPositive(struct Time t); + +/// @brief Return the sign of a Time. +/// @param t The Time to check. +/// @returns 0 if all units equal the lower bound, -1 if any unit is less than +/// the lower bound, or 1 if any unit is greater than the lower bound. +/// @note The Time must be normalized to yield an acurate result. +int GetTimeSign(struct Time t); + +/// @brief Create a new calendar Time. +/// @param nMinsPerHour The number of minutes per hour (1..60). If 0, will use +/// the module's default setting. +struct Time NewTime(int nMinsPerHour = 0); + +/// @brief Create a new duration Time. +/// @param nMinsPerHour The number of minutes per hour (1..60). If 0, will use +/// the module's default setting. +struct Time NewDuration(int nMinsPerHour = 0); + +/// @brief Convert a calendar Time into a duration Time. +/// @note This is safe to call on a duration Time. +struct Time TimeToDuration(struct Time t); + +/// @brief Convert a duration Time into a calendar Time. +/// @note This is safe to call on a calendar Time. +struct Time DurationToTime(struct Time d); + +/// @brief Distribute units in a Time, optionally converting minutes per hour. +/// @details Units that overflow their range have the excess added to the next +/// highest unit (e.g., 1500 msec -> 1 sec, 500 msec). If `nMinsPerHour` +/// does not match `t.MinsPerHour`, the minutes, seconds, and milliseconds +/// will be recalculated to match the new setting. +/// @param t The Time to normalize. +/// @param nMinsPerHour The number of minutes per hour to normalize with. If 0, +/// will use `t.MinsPerHour`. +/// @note If `t` is a duration Time, all non-zero units will be either positive +/// or negative (i.e., not a mix of both). +/// @note If `t` is a calendar Time and any unit in `t` falls outside the bounds +/// after normalization, an invalid Time is returned. You can check for this +/// using GetIsTimeValid(). +struct Time NormalizeTime(struct Time t, int nMinsPerHour = 0); + +/// @brief Check if any unit in a normalized time is outside its range. +/// @param t The Time to validate. +/// @param bNormalize Whether to normalize the time before checking. You should +/// only set this to FALSE if you know `t` is already normalized and want to +/// save cycles. +/// @returns TRUE if valid, FALSE otherwise. +int GetIsTimeValid(struct Time t, int bNormalize = TRUE); + +/// @brief Create a duration Time, representing an amount of time. +/// @note All units count from 0. Negative numbers are allowed. +/// @param nYear The number of years (0..32000). +/// @param nMonth The number of month (0..11). +/// @param nDay The number of day (0..27). +/// @param nHour The number of hours (0..23). +/// @param nMinute The number of minutes (0..nMinsPerHour). +/// @param nSecond The number of seconds (0..59). +/// @param nMillisecond The number of milliseconds (0..999). +/// @param nMinsPerHour The number of minutes per hour (1..60). If 0, will use +/// the module's default setting. +/// @returns A normalized duration Time. +struct Time GetDuration(int nYears = 0, int nMonths = 0, int nDays = 0, int nHours = 0, int nMinutes = 0, int nSeconds = 0, int nMilliseconds = 0, int nMinsPerHour = 0); + +/// @brief Create a calendar Time, representing a moment in time. +/// @param nYear The year (0..32000). +/// @param nMonth The month of the year (1..12). +/// @param nDay The day of the month (1..28). +/// @param nHour The hour (0..23). +/// @param nMinute The minute (0..nMinsPerHour). +/// @param nSecond The second (0..59). +/// @param nMillisecond The millisecond (0..999). +/// @param nMinsPerHour The number of minutes per hour (1..60). If 0, will use +/// the module's default setting. +/// @returns A normalized calendar Time. +struct Time GetTime(int nYear = 0, int nMonth = 1, int nDay = 1, int nHour = 0, int nMinute = 0, int nSecond = 0, int nMillisecond = 0, int nMinsPerHour = 0); + +/// @brief Convert a Time to an in-game time (i.e., 60 minutes per hour). +/// @param t The Time to convert. +/// @note Alias for NormalizeTime(t, 60). +struct Time TimeToGameTime(struct Time t); + +/// @brief Convert an in-game time (i.e., 60 minutes per hour) to a Time. +/// @param t The Time to convert. +/// @note Alias for NormalizeTime(t, HoursToMinutes()). +struct Time GameTimeToTime(struct Time t); + +/// @brief Add a Time to another. +/// @param a The Time to modify. +/// @param b The Time to add. +/// @returns A Time of the same type and minutes per hour as `a`. +/// @note You can safely mix calendar or duration Times, as well as Times with +/// different minutes per hour settings. +struct Time AddTime(struct Time a, struct Time b); + +/// @brief Subtract a Time from another. +/// @param a The Time to modify. +/// @param b The Time to subtract. +/// @returns A Time of the same type and minutes per hour as `a`. +/// @note You can safely mix calendar or duration Times, as well as Times with +/// different minutes per hour settings. +struct Time SubtractTime(struct Time a, struct Time b); + +/// @brief Get the current calendar date and clock time as a calendar Time. +/// @note A calendar Time with a `MinsPerHour` matching to the module's setting. +struct Time GetCurrentTime(); + +/// @brief Get the current calendar date and in-game time as a calendar Time. +/// @returns A calendar Time with a `MinsPerHour` of 60. +/// @note Alias for TimeToGameTime(GetCurrentTime()). +struct Time GetCurrentGameTime(); + +/// @brief Set the current calendar date and clock time. +/// @param t The time to set the calendar and clock to. Must be a valid calendar +/// Time that is after the current time. +void SetCurrentTime(struct Time t); + +/// @brief Set the current calendar date and clock time forwards. +/// @param d A duration Time by which to advance the time. Must be positive. +void AdvanceCurrentTime(struct Time d); + +/// @brief Drop smaller units from a Time. +/// @param t The Time to modify. +/// @param nUnit A TIME_UNIT_* constant representing the maximum precision. +/// Units more precise than this are set to their lowest value. +struct Time GetPrecisionTime(struct Time t, int nUnit); + +/// @brief Get the duration of the interval between two Times. +/// @param a The calendar Time at the start of interval. +/// @param b The calendar Time at the end of the interval. +/// @returns A normalized duration Time. The duration will be negative if a is +/// after b and positive if b is after a. If the times are equivalent, the +/// duration will equal 0. +struct Time GetDurationBetween(struct Time tStart, struct Time tEnd); + +/// @brief Get the duration of the interval between a Time and the current time. +/// @param tSince The Time at the start of the interval. +/// @returns A normalized duration Time. The duration will be negative if a is +/// after b and positive if b is after a. If the times are equivalent, the +/// duration will equal 0. +struct Time GetDurationSince(struct Time tSince); + +/// @brief Get the duration of the interval between the current time and a Time. +/// @param tUntil The Time at the end of the interval. +/// @returns A normalized duration Time. The duration will be negative if a is +/// after b and positive if b is after a. If the times are equivalent, the +/// duration will equal 0. +struct Time GetDurationUntil(struct Time tUntil); + +/// @brief Compare two Times and find which is later. +/// @param a The Time to check. +/// @param b The Time to check against. +/// @returns 0 if a == b, -1 if a < b, and 1 if a > b. +int CompareTime(struct Time a, struct Time b); + +/// @brief Check whether a Time is after another Time. +/// @param a The Time to check. +/// @param b The Time to check against. +/// @returns TRUE if a is after b, FALSE otherwise +int GetIsTimeAfter(struct Time a, struct Time b); + +/// @brief Check whether a Time is before another Time. +/// @param a The Time to check. +/// @param b The Time to check against. +/// @returns TRUE if a is before b, FALSE otherwise +int GetIsTimeBefore(struct Time a, struct Time b); + +/// @brief Check whether a Time is equal to another Time. +/// @param a The Time to check. +/// @param b The Time to check against. +/// @returns TRUE if a is equivalent to b, FALSE otherwise. +/// @note This checks if the normalized Times represent equal moments in time. +/// If you want to instead check if two Time structs are exactly equal, use +/// `a == b`. +int GetIsTimeEqual(struct Time a, struct Time b); + +// ----- Float Conversion ------------------------------------------------------ + +/// @brief Convert years to seconds. +/// @param nYears The number of years to convert. +float Years(int nYears); + +/// @brief Convert months to seconds. +/// @param nMonths The number of months to convert. +float Months(int nMonths); + +/// @brief Convert days to seconds. +/// @param nDays The number of days to convert. +float Days(int nDays); + +/// @brief Convert hours to seconds. +/// @param nHours The number of hours to convert. +float Hours(int nHours); + +/// @brief Convert minutes to seconds. +/// @param nMinutes The number of minutes to convert. +float Minutes(int nMinutes); + +/// @brief Convert seconds to seconds. +/// @param nSeconds The number of seconds to convert. +float Seconds(int nSeconds); + +/// @brief Convert milliseconds to seconds. +/// @param nYears The number of milliseconds to convert. +float Milliseconds(int nMilliseconds); + +/// @brief Convert a duration Time to a float. +/// @param d The duration Time to convert. +/// @returns A float representing the number of seconds in `t`. Always has a +/// minutes per hour setting equal to the module's. +/// @note Use this function to pass a Time to a function like DelayCommand(). +/// @note Long durations may lose precision when converting. Use with caution. +float DurationToFloat(struct Time d); + +/// @brief Convert a float to a duration Time. +/// @param fDur A float representing a number of seconds. +/// @returns A duration Time with a minutes per hour setting equal to the +/// module's. +struct Time FloatToDuration(float fDur); + +// ----- Json Conversion ------------------------------------------------------- + +/// @brief Convert a Time into a json object. +/// @details The json object will have a key for each field of the Time struct. +/// Since this includes the minutes per hour setting, this object is safe to +/// be stored in a database if it is possible the module's minutes per hour +/// setting will change. The object can be converted back using +/// JsonToTime(). +/// @param t The Time to convert +json TimeToJson(struct Time t); + +/// @brief Convert a json object into a Time. +/// @param j The json object to convert. +struct Time JsonToTime(json j); + +// ----- Local Variables ------------------------------------------------------- + +/// @brief Return a Time from a local variable. +/// @param oObject The object to get the local variable from. +/// @param sVarName The varname for the local variable. +struct Time GetLocalTime(object oObject, string sVarName); + +/// @brief Store a Time as a local variable. +/// @param oObject The object to store the local variable on. +/// @param sVarName The varname for the local variable. +/// @param tValue The Time to store. +void SetLocalTime(object oObject, string sVarName, struct Time tValue); + +/// @brief Delete a Time from a local variable. +/// @param oObject The object to delete the local variable from. +/// @param sVarName The varname for the local variable. +void DeleteLocalTime(object oObject, string sVarName); + +// ----- String Conversions ---------------------------------------------------- + +/// @brief Convert a Time into a string. +/// @param t The Time to convert. +/// @param bNormalize Whether to normalize the Time before converting. +/// @returns An ISO 8601 formatted datetime, e.g., "1372-06-01 13:00:00:000". +/// @note If `t` is a duration Time and is negative, the returned value will be +/// preceded by a `-` character. +string TimeToString(struct Time t, int bNormalize = TRUE); + +/// @brief Convert an ISO 8601 formatted datetime string into a calendar Time. +/// @param sTime The string to convert. +/// @param nMinsPerHour The number of minutes per hour expected in the Time. If +/// 0, will use the module setting. +/// @note The returned Time is not normalized. +/// @note If the first character in `sTime` is a `-`, all values will be treated +/// as negative. This will make the returned Time invalid when normalized. +struct Time StringToTime(string sTime, int nMinsPerHour = 0); + +/// @brief Convert an ISO 8601 formatted datetime string into a duration Time. +/// @param sTime The string to convert. +/// @param nMinsPerHour The number of minutes per hour expected in the Time. If +/// 0, will use the module setting. +/// @note The returned Time is not normalized. +/// @note If the first character in `sTime` is a `-`, all values will be treated +/// as negative. +struct Time StringToDuration(string sTime, int nMinsPerHour = 0); + +// ----------------------------------------------------------------------------- +// Function Definitions +// ----------------------------------------------------------------------------- + +int HoursToMinutes(int nHours = 1) +{ + return FloatToInt(HoursToSeconds(nHours)) / 60; +} + +// ----- Times ----------------------------------------------------------------- + +int GetAnyTimeUnitNegative(struct Time t) +{ + return t.Year < 0 || t.Month < t.Type || t.Day < t.Type || + t.Hour < 0 || t.Minute < 0 || t.Second < 0 || t.Millisecond < 0; +} + +int GetAnyTimeUnitPositive(struct Time t) +{ + return t.Year > 0 || t.Month > t.Type || t.Day > t.Type || + t.Hour > 0 || t. Minute > 0 || t.Second > 0 || t.Millisecond > 0; +} + +int GetTimeSign(struct Time t) +{ + return GetAnyTimeUnitNegative(t) ? -1 : GetAnyTimeUnitPositive(t) ? 1 : 0; +} + +struct Time NewTime(int nMinsPerHour = 0) +{ + struct Time t; + t.Type = TIME_TYPE_CALENDAR; + t.MinsPerHour = nMinsPerHour <= 0 ? HoursToMinutes() : clamp(nMinsPerHour, 1, 60); + t.Month = 1; + t.Day = 1; + return t; +} + +struct Time NewDuration(int nMinsPerHour = 0) +{ + struct Time t; + t.Type = TIME_TYPE_DURATION; + t.MinsPerHour = nMinsPerHour <= 0 ? HoursToMinutes() : clamp(nMinsPerHour, 1, 60); + return t; +} + +struct Time TimeToDuration(struct Time t) +{ + t.Day -= t.Type; + t.Month -= t.Type; + t.Type = TIME_TYPE_DURATION; + return t; +} + +struct Time DurationToTime(struct Time d) +{ + d.Day += (1 - d.Type); + d.Month += (1 - d.Type); + d.Type = TIME_TYPE_CALENDAR; + return d; +} + +struct Time NormalizeTime(struct Time t, int nMinsPerHour = 0) +{ + // Convert everything to a duration for ease of calculation + int nType = t.Type; + t = TimeToDuration(t); + + // If the conversion factor was not set, we assume it's using the module's. + if (t.MinsPerHour <= 0) + t.MinsPerHour = HoursToMinutes(); + + // If this is > 0, we will adjust the time's conversion factor to match the + // requested value. Otherwise, assume we're using the same conversion factor + // and just prettifying units. + nMinsPerHour = nMinsPerHour > 0 ? clamp(nMinsPerHour, 1, 60) : t.MinsPerHour; + + if (t.MinsPerHour != nMinsPerHour) + { + // Convert everything to milliseconds so we don't lose precision when + // converting to a smaller mins-per-hour. + t.Millisecond += (t.Minute * 60 + t.Second) * 1000; + t.Millisecond = t.Millisecond * nMinsPerHour / t.MinsPerHour; + t.Second = 0; + t.Minute = 0; + t.MinsPerHour = nMinsPerHour; + } + + // Distribute units. + int nFactor; + if (abs(t.Millisecond) >= (nFactor = 1000)) + { + t.Second += t.Millisecond / nFactor; + t.Millisecond %= nFactor; + } + + if (abs(t.Second) >= (nFactor = 60)) + { + t.Minute += t.Second / nFactor; + t.Second %= nFactor; + } + + if (abs(t.Minute) >= (nFactor = t.MinsPerHour)) + { + t.Hour += t.Minute / nFactor; + t.Minute %= nFactor; + } + + if (abs(t.Hour) >= (nFactor = 24)) + { + t.Day += t.Hour / nFactor; + t.Hour %= nFactor; + } + + if (abs(t.Day) >= (nFactor = 28)) + { + t.Month += t.Day / nFactor; + t.Day %= nFactor; + } + + if (abs(t.Month) >= (nFactor = 12)) + { + t.Year += t.Month / nFactor; + t.Month %= nFactor; + } + + // A mix of negative and positive units means we need to consolidate and + // re-normalize. + if (GetAnyTimeUnitPositive(t) && GetAnyTimeUnitNegative(t)) + { + struct Time d = NewDuration(t.MinsPerHour); + d.Millisecond = (t.Minute * 60 + t.Second) * 1000 + t.Millisecond; + d.Hour = ((t.Year * 12 + t.Month) * 28 + t.Day) * 24 + t.Hour; + + // If that didn't fix it, borrow a unit + if ((d.Millisecond >= 0) != (d.Hour >= 0)) + { + d.Millisecond += sign(d.Hour) * 1000 * 60 * d.MinsPerHour; + d.Hour -= sign(d.Hour); + } + + t = NormalizeTime(d); + } + + // Convert back to a calendar Time if needed. + if (nType) + { + if (GetAnyTimeUnitNegative(t)) + return TIME_INVALID; + + return DurationToTime(t); + } + + return t; +} + +int GetIsTimeValid(struct Time t, int bNormalize = TRUE) +{ + if (bNormalize) + t = NormalizeTime(t); + return t != TIME_INVALID; +} + +struct Time GetDuration(int nYears = 0, int nMonths = 0, int nDays = 0, int nHours = 0, int nMinutes = 0, int nSeconds = 0, int nMilliseconds = 0, int nMinsPerHour = 0) +{ + struct Time d = NewDuration(nMinsPerHour); + d.Year = nYears; + d.Month = nMonths; + d.Day = nDays; + d.Hour = nHours; + d.Minute = nMinutes; + d.Second = nSeconds; + d.Millisecond = nMilliseconds; + return NormalizeTime(d); +} + +struct Time GetTime(int nYear = 0, int nMonth = 1, int nDay = 1, int nHour = 0, int nMinute = 0, int nSecond = 0, int nMillisecond = 0, int nMinsPerHour = 0) +{ + struct Time t = NewTime(nMinsPerHour); + t.Year = nYear; + t.Month = nMonth; + t.Day = nDay; + t.Hour = nHour; + t.Minute = nMinute; + t.Second = nSecond; + t.Millisecond = nMillisecond; + return NormalizeTime(t); +} + +struct Time TimeToGameTime(struct Time t) +{ + return NormalizeTime(t, 60); +} + +struct Time GameTimeToTime(struct Time t) +{ + return NormalizeTime(t, HoursToMinutes()); +} + +struct Time AddTime(struct Time a, struct Time b) +{ + // Convert everything to a duration to ensure even comparison + int nType = a.Type; + a = NormalizeTime(TimeToDuration(a)); + b = NormalizeTime(TimeToDuration(b), a.MinsPerHour); + + a.Year += b.Year; + a.Month += b.Month; + a.Day += b.Day; + a.Hour += b.Hour; + a.Minute += b.Minute; + a.Second += b.Second; + a.Millisecond += b.Millisecond; + + // Convert back to calendar time if needed + if (nType) + a = DurationToTime(a); + + return NormalizeTime(a); +} + +struct Time SubtractTime(struct Time a, struct Time b) +{ + // Convert everything to a duration to ensure even comparison + int nType = a.Type; + a = NormalizeTime(TimeToDuration(a)); + b = NormalizeTime(TimeToDuration(b), a.MinsPerHour); + + a.Year -= b.Year; + a.Month -= b.Month; + a.Day -= b.Day; + a.Hour -= b.Hour; + a.Minute -= b.Minute; + a.Second -= b.Second; + a.Millisecond -= b.Millisecond; + + // Convert back to calendar time if needed + if (nType) + a = DurationToTime(a); + + return NormalizeTime(a); +} + +struct Time GetCurrentTime() +{ + struct Time t = NewTime(); + t.Year = GetCalendarYear(); + t.Month = GetCalendarMonth(); + t.Day = GetCalendarDay(); + t.Hour = GetTimeHour(); + t.Minute = GetTimeMinute(); + t.Second = GetTimeSecond(); + t.Millisecond = GetTimeMillisecond(); + return t; +} + +struct Time GetCurrentGameTime() +{ + return TimeToGameTime(GetCurrentTime()); +} + +void SetCurrentTime(struct Time t) +{ + t = NormalizeTime(t, HoursToMinutes()); + struct Time tCurrent = GetCurrentTime(); + if (GetIsTimeAfter(t, tCurrent)) + { + SetTime(t.Hour, t.Minute, t.Second, t.Millisecond); + SetCalendar(t.Year, t.Month, t.Day); + } + else + { + CriticalError("Cannot set time to " + TimeToString(t, FALSE) + " " + + "because it is before " + TimeToString(tCurrent)); + } +} + +void AdvanceCurrentTime(struct Time d) +{ + int nSign = GetTimeSign(d); + if (nSign > 0) + { + d = AddTime(GetCurrentTime(), d); + SetTime(d.Hour, d.Minute, d.Second, d.Millisecond); + SetCalendar(d.Year, d.Month, d.Day); + } + else if (nSign < 0) + CriticalError("Cannot advance time by a negative amount"); +} + +struct Time GetPrecisionTime(struct Time t, int nUnit) +{ + while (nUnit < TIME_UNIT_MILLISECOND) + { + switch (++nUnit) + { + case TIME_UNIT_YEAR: t.Year = 0; break; + case TIME_UNIT_MONTH: t.Month = 0 + t.Type; break; + case TIME_UNIT_DAY: t.Day = 0 + t.Type; break; + case TIME_UNIT_HOUR: t.Hour = 0; break; + case TIME_UNIT_MINUTE: t.Minute = 0; break; + case TIME_UNIT_SECOND: t.Second = 0; break; + case TIME_UNIT_MILLISECOND: t.Millisecond = 0; break; + } + } + + return t; +} + +struct Time GetDurationBetween(struct Time tStart, struct Time tEnd) +{ + // Convert to duration before passing to ensure we get a duration back + return SubtractTime(TimeToDuration(tEnd), tStart); +} + +struct Time GetDurationSince(struct Time tSince) +{ + return GetDurationBetween(tSince, GetCurrentTime()); +} + +struct Time GetDurationUntil(struct Time tUntil) +{ + return GetDurationBetween(tUntil, GetCurrentTime()); +} + +int CompareTime(struct Time a, struct Time b) +{ + return GetTimeSign(GetDurationBetween(b, a)); +} + +int GetIsTimeAfter(struct Time a, struct Time b) +{ + return CompareTime(a, b) > 0; +} + +int GetIsTimeBefore(struct Time a, struct Time b) +{ + return CompareTime(a, b) < 0; +} + +int GetIsTimeEqual(struct Time a, struct Time b) +{ + return !CompareTime(a, b); +} + +// ----- Float Conversion ------------------------------------------------------ + +float Years(int nYears) +{ + return HoursToSeconds(nYears * 12 * 28 * 24); +} + +float Months(int nMonths) +{ + return HoursToSeconds(nMonths * 28 * 24); +} + +float Days(int nDays) +{ + return HoursToSeconds(nDays * 24); +} + +float Hours(int nHours) +{ + return HoursToSeconds(nHours); +} + +float Minutes(int nMinutes) +{ + return nMinutes * 60.0; +} + +float Seconds(int nSeconds) +{ + return IntToFloat(nSeconds); +} + +float Milliseconds(int nMilliseconds) +{ + return nMilliseconds / 1000.0; +} + +float DurationToFloat(struct Time d) +{ + d = NormalizeTime(TimeToDuration(d), HoursToMinutes()); + return Years(d.Year) + Months(d.Month) + Days(d.Day) + Hours(d.Hour) + + Minutes(d.Minute) + Seconds(d.Second) + Milliseconds(d.Millisecond); +} + +struct Time FloatToDuration(float fDur) +{ + struct Time t = NewDuration(HoursToMinutes()); + t.Millisecond = FloatToInt(frac(fDur) * 1000); + t.Second = FloatToInt(fmod(fDur, 60.0)); + t.Minute = FloatToInt(fDur / 60) % t.MinsPerHour; + int nHours = FloatToInt(fDur / HoursToSeconds(1)); + t.Hour = nHours % 24; + t.Day = (nHours / 24) % 28; + t.Month = (nHours / 24 / 28) % 12; + t.Year = (nHours / 24 / 28 / 12); + return t; +} + +// ----- Json Conversion ------------------------------------------------------- + +json TimeToJson(struct Time t) +{ + json j = JsonObject(); + j = JsonObjectSet(j, TIME_TYPE, JsonInt(t.Type)); + j = JsonObjectSet(j, TIME_YEAR, JsonInt(t.Year)); + j = JsonObjectSet(j, TIME_MONTH, JsonInt(t.Month)); + j = JsonObjectSet(j, TIME_DAY, JsonInt(t.Day)); + j = JsonObjectSet(j, TIME_HOUR, JsonInt(t.Hour)); + j = JsonObjectSet(j, TIME_MINUTE, JsonInt(t.Minute)); + j = JsonObjectSet(j, TIME_SECOND, JsonInt(t.Second)); + j = JsonObjectSet(j, TIME_MILLISECOND, JsonInt(t.Millisecond)); + j = JsonObjectSet(j, TIME_MINSPERHOUR, JsonInt(t.MinsPerHour)); + return j; +} + +struct Time JsonToTime(json j) +{ + if (JsonGetType(j) != JSON_TYPE_OBJECT) + return TIME_INVALID; + + struct Time t; + t.Type = JsonGetInt(JsonObjectGet(j, TIME_TYPE)); + t.Year = JsonGetInt(JsonObjectGet(j, TIME_YEAR)); + t.Month = JsonGetInt(JsonObjectGet(j, TIME_MONTH)); + t.Day = JsonGetInt(JsonObjectGet(j, TIME_DAY)); + t.Hour = JsonGetInt(JsonObjectGet(j, TIME_HOUR)); + t.Minute = JsonGetInt(JsonObjectGet(j, TIME_MINUTE)); + t.Second = JsonGetInt(JsonObjectGet(j, TIME_SECOND)); + t.Millisecond = JsonGetInt(JsonObjectGet(j, TIME_MILLISECOND)); + t.MinsPerHour = JsonGetInt(JsonObjectGet(j, TIME_MINSPERHOUR)); + return t; +} + +// ----- Local Variables ------------------------------------------------------- + +struct Time GetLocalTime(object oObject, string sVarName) +{ + return JsonToTime(GetLocalJson(oObject, TIME_PREFIX + sVarName)); +} + +void SetLocalTime(object oObject, string sVarName, struct Time tValue) +{ + SetLocalJson(oObject, TIME_PREFIX + sVarName, TimeToJson(tValue)); +} + +void DeleteLocalTime(object oObject, string sVarName) +{ + DeleteLocalJson(oObject, TIME_PREFIX + sVarName); +} + +// ----- String Conversions ---------------------------------------------------- + +string TimeToString(struct Time t, int bNormalize = TRUE) +{ + if (bNormalize) + t = NormalizeTime(t); + + json j = JsonArray(); + j = JsonArrayInsert(j, JsonString(t.Type || GetTimeSign(t) >= 0 ? "" : "-")); + j = JsonArrayInsert(j, JsonInt(abs(t.Year))); + j = JsonArrayInsert(j, JsonInt(abs(t.Month))); + j = JsonArrayInsert(j, JsonInt(abs(t.Day))); + j = JsonArrayInsert(j, JsonInt(abs(t.Hour))); + j = JsonArrayInsert(j, JsonInt(abs(t.Minute))); + j = JsonArrayInsert(j, JsonInt(abs(t.Second))); + j = JsonArrayInsert(j, JsonInt(abs(t.Millisecond))); + return FormatValues(j, "%s%04d-%02d-%02d %02d:%02d:%02d:%03d"); +} + +struct Time _StringToTime(string sTime, struct Time t) +{ + if (sTime == "") + return TIME_INVALID; + + string sDelims = "-- :::"; + int nUnit = TIME_UNIT_YEAR; + int nPos, nLength = GetStringLength(sTime); + int nSign = 1; + + // Strip off an initial "-" + if (GetChar(sTime, 0) == "-") + { + nSign = -1; + nPos++; + } + + while (nPos < nLength) + { + string sDelim, sToken, sChar; + while (HasSubString(CHARSET_NUMERIC, (sChar = GetChar(sTime, nPos++)))) + sToken += sChar; + + if (GetStringLength(sToken) < 1) + return TIME_INVALID; + + // If the first character was a -, all subsequent values are negative + int nToken = StringToInt(sToken) * nSign; + switch (nUnit) + { + case TIME_UNIT_YEAR: t.Year = nToken; break; + case TIME_UNIT_MONTH: t.Month = nToken; break; + case TIME_UNIT_DAY: t.Day = nToken; break; + case TIME_UNIT_HOUR: t.Hour = nToken; break; + case TIME_UNIT_MINUTE: t.Minute = nToken; break; + case TIME_UNIT_SECOND: t.Second = nToken; break; + case TIME_UNIT_MILLISECOND: t.Millisecond = nToken; break; + default: + return TIME_INVALID; + } + + // Check if we encountered a delimiter with no characters following. + if (nPos == nLength && sChar != "") + return TIME_INVALID; + + // If we run out of characters before we've parsed all the units, we can + // return the partial time. However, if we run into an unexpected + // character, we should yield an invalid time. + if (sChar != GetChar(sDelims, nUnit++)) + { + if (sChar == "") + return t; + return TIME_INVALID; + } + } + + return t; +} + +struct Time StringToTime(string sTime, int nMinsPerHour = 0) +{ + return _StringToTime(sTime, NewTime(nMinsPerHour)); +} + +struct Time StringToDuration(string sTime, int nMinsPerHour = 0) +{ + return _StringToTime(sTime, NewDuration(nMinsPerHour)); +} diff --git a/_module/nss/util_i_unittest.nss b/_module/nss/util_i_unittest.nss new file mode 100644 index 0000000..136c464 --- /dev/null +++ b/_module/nss/util_i_unittest.nss @@ -0,0 +1,375 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_unittest.nss +/// @author Ed Burke (tinygiant98) +/// @brief Functions for managing unit test reporting. +/// ---------------------------------------------------------------------------- +/// @details +/// +/// Variable Conventions: +/// +/// Tests can be written in just about any format, however since tests tend to be +/// repetitive, having a variable and formatting convention can make building +/// multiple tests quick and easy. Following are variable naming conventions +/// and an example that showcases how to use them. +/// +/// Variable Naming: +/// ix - Function Input Variables +/// ex - Expected Function Result Variables +/// rx - Actual Function Result Variables +/// bx - Boolean Test Result Variables +/// tx - Timer Variables +/// +/// Convenience Functions: +/// _i : IntToString +/// _f : FloatToString; Rounds to significant digits +/// _b : Returns `True` or `False` (literals) +/// +/// _q : Returns string wrapped in single quotes +/// _qq : Returns string wrapped in double quotes +/// _p : Returns string wrapped in parenthesis +/// +/// Timers: +/// To start a timer: +/// t1 = Timer(); : Sets timer variable `t1` to GetMicrosecondCounter() +/// +/// To end a timer and save the results: +/// t1 = Timer(t1); : Sets timer variable `t1` to GetMicrosecondCounter() - t1 +/// +/// The following example shows how to create a grouped assertion and display +/// only relevant results, assuming only assertion failures are of +/// interest. If you always want to see expanded results regardless of test +/// outcome, set UNITTEST_ALWAYS_EXPAND to TRUE in `util_c_unittest`. +/// +/// For example purposes only, this unit test sample code will run a unittest +/// against the following function, which will return: +/// -1, if n <= 0 +/// 20 * n, if 0 < n <= 3 +/// 100, if n > 3 +/// +/// ```nwscript +/// int unittest_demo_ConvertValue(int n) +/// { +/// return n <= 0 ? -1 : n > 3 ? 100 : 20 * n; +/// } +/// ``` +/// +/// The following unit test will run against the function above for three test cases: +/// - Out of bounds (low) -> n <= 0; +/// - In bounds -> 0 < n <= 3; +/// - Out of bounds (high) -> n > 3; +/// +/// ```nwscript +/// int unittest_ConvertValue() +/// { +/// int i1, i2, i3; +/// int e1, e2, e3; +/// int r1, r2, r3; +/// int b1, b2, b3, b; +/// int t1, t2, t3, t; +/// +/// // Setup the input values +/// i1 = -10; +/// i2 = 2; +/// i3 = 12; +/// +/// // Setup the expected return values +/// e1 = -1; +/// e2 = 40; +/// e3 = 100; +/// +/// // Run the unit tests with timers +/// t = Timer(); +/// t1 = Timer(); r1 = unittest_demo_ConvertValue(i1); t1 = Timer(t1); +/// t2 = Timer(); r2 = unittest_demo_ConvertValue(i2); t2 = Timer(t2); +/// t3 = Timer(); r3 = unittest_demo_ConvertValue(i3); t3 = Timer(t3); +/// t = Timer(t); +/// +/// // Populate the results +/// b = (b1 = r1 == e1) & +/// (b2 = r2 == e2) & +/// (b3 = r3 == e3); +/// +/// // Display the result +/// if (!AssertGroup("ConvertValue()", b)) +/// { +/// if (!Assert("Out of bounds (low)", b1)) +/// DescribeTestParameters(_i(i1), _i(e1), _i(r1)); +/// DescribeTestTime(t1); +/// +/// if (!Assert("In bounds", b2)) +/// DescribeTestParameters(_i(i2), _i(e2), _i(r2)); +/// DescribeTestTime(t2); +/// +/// if (!Assert("Out of bounds (high)", b3)) +/// DescribeTestParameters(_i(i3), _i(e3), _i(r3)); +/// DescribeTestTime(t3); +/// } DescribeGroupTime(t); Outdent(); +/// } +/// Note: Use of ResetIndent() or another indentation function, such as +/// Outdent(), may be required if moving to another group assertion. + +#include "util_c_unittest" +#include "util_i_strings" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +string TEST_INDENT = "TEST_INDENT"; + +string TEST_PASS = HexColorString("PASS", COLOR_GREEN_LIGHT); +string TEST_FAIL = HexColorString("FAIL", COLOR_RED_LIGHT); +string TEST_DELIMITER = HexColorString(" | ", COLOR_WHITE); + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Establishes or calculates a timer or elapsed value. +/// @param t Previous timer value derived from this function. +/// @note Calling this function without parameter `t` specified will +/// return a starting value in microseconds. When the code in +/// question has been run, call this function again and pass +/// the previously returned value as parameter `t` to calculate +/// the total elapsed time for between calls to this function. +int Timer(int t = 0); + +/// @brief Reset the indentation level used in displaying test results. +/// @returns The indenation string used to pad test result output. +string ResetIndent(); + +/// @brief Indent test results display by one indentation level. +/// @param bReset If TRUE, will reset the indentation level to 0 before +/// adding an indentation level. +/// @returns The indenation string used to pad test result output. +string Indent(int bReset = FALSE); + +/// @brief Outdent test results display by one indentation level. +/// @returns The indenation string used to pad test result output. +string Outdent(); + +/// @brief Provide a test suite description. +/// @param sDescription The description to display. +/// @note Test suite description will always display at indentation level +/// 0 and will reset the indentation level for the subsequest assertions. +void DescribeTestSuite(string sDescription); + +/// @brief Provide a test group description. +/// @param sDescription The description to display. +/// @note Test groups are used to minimize unit test output if all tests +/// within a group pass. This function only provides a header for the +/// test group. To provide a test group description combined with +/// test group ouput, use AssertGroup(). +void DescribeTestGroup(string sDescription); + +/// @brief Display the parameter used in a test. +/// @param sInput The input data. +/// @param sExpected The expected test result. +/// @param sReceived The actual test result. +/// @note Each paramater is optional. If any parameter is an empty string, +/// that parameter will not be output. +void DescribeTestParameters(string sInput = "", string sExpected = "", string sReceived = ""); + +/// @brief Display function timer result. +/// @param nTime Function timer result, in microseconds. +/// @note This function is intended to use output from GetMicrosecondCounter(). +void DescribeTestTime(int nTime); + +/// @brief Display function timer result. +/// @param nTime Function timer result, in microseconds. +/// @note This function is intended to use output from GetMicrosecondCounter(). +void DescribeGroupTime(int nTime); + +/// @brief Display the results of a unit test. +/// @param sTest The name of the unit test. +/// @param bAssertion The results of the unit test. +/// @returns The results of the unit test. +int Assert(string sTest, int bAssertion); + +/// @brief Display the results of a group test. +/// @param sTest The name of the group test. +/// @param bAssertion The results of the group test. +/// @returns The results of the group test. +int AssertGroup(string sGroup, int bAssertion); + +// ----------------------------------------------------------------------------- +// Private Function Implementations +// ----------------------------------------------------------------------------- + +string _GetIndent(int bReset = FALSE) +{ + if (bReset) + ResetIndent(); + + string sIndent; + int nIndent = GetLocalInt(GetModule(), TEST_INDENT); + if (nIndent == 0) + return ""; + + while (nIndent-- > 0) + sIndent += " "; + + return sIndent; +} + +// ----------------------------------------------------------------------------- +// Public Function Implementations +// ----------------------------------------------------------------------------- + +string _i(int n) { return IntToString(n); } +string _f(float f) { return FormatFloat(f, "%!f"); } +string _b(int b) { return b ? "True" : "False"; } + +string _q(string s) { return "'" + s + "'"; } +string _qq(string s) { return "\"" + s + "\""; } +string _p(string s) { return "(" + s + ")"; } + +int Timer(int t = 0) +{ + return GetMicrosecondCounter() - t; +} + +string ResetIndent() +{ + DeleteLocalInt(GetModule(), TEST_INDENT); + return _GetIndent(); +} + +string Indent(int bReset = FALSE) +{ + if (bReset) + ResetIndent(); + + int nIndent = GetLocalInt(GetModule(), TEST_INDENT); + SetLocalInt(GetModule(), TEST_INDENT, ++nIndent); + return _GetIndent(); +} + +string Outdent() +{ + int nIndent = GetLocalInt(GetModule(), TEST_INDENT); + SetLocalInt(GetModule(), TEST_INDENT, max(0, --nIndent)); + return _GetIndent(); +} + +void DescribeTestSuite(string sDescription) +{ + sDescription = HexColorString("Test Suite ", UNITTEST_TITLE_COLOR) + + HexColorString(sDescription, UNITTEST_NAME_COLOR); + Indent(TRUE); + HandleUnitTestOutput(sDescription); +} + +void DescribeTestGroup(string sDescription) +{ + sDescription = HexColorString("Test Group ", UNITTEST_TITLE_COLOR) + + HexColorString(sDescription, UNITTEST_NAME_COLOR); + HandleUnitTestOutput(_GetIndent() + sDescription); + Indent(); +} + +void DescribeTestParameters(string sInput, string sExpected, string sReceived) +{ + Indent(); + if (sInput != "") + { + json jInput = JsonParse(sInput); + if (jInput != JSON_NULL && JsonGetLength(jInput) > 0) + { + if (JsonGetType(jInput) == JSON_TYPE_ARRAY) + { + string s = "WITH atoms AS (SELECT atom FROM json_each(@json)) " + + "SELECT group_concat(atom, ' | ') FROM atoms;"; + sqlquery q = SqlPrepareQueryObject(GetModule(), s); + SqlBindJson(q, "@json", jInput); + sInput = SqlStep(q) ? SqlGetString(q, 0) : sInput; + } + else if (JsonGetType(jInput) == JSON_TYPE_OBJECT) + { + string s = "WITH kvps AS (SELECT key, value FROM json_each(@json)) " + + "SELECT group_concat(key || ' = ' || (IFNULL(value, '\"\"\"\"')), ' | ') FROM kvps;"; + sqlquery q = SqlPrepareQueryObject(GetModule(), s); + SqlBindJson(q, "@json", jInput); + sInput = SqlStep(q) ? SqlGetString(q, 0) : sInput; + } + + sInput = RegExpReplace("(?:^|\\| )(.*?)(?= =)", sInput, HexToColor(COLOR_BLUE_STEEL) + "$&"); + sInput = RegExpReplace("\\||=", sInput, HexToColor(COLOR_WHITE) + "$&"); + } + + sInput = _GetIndent() + HexColorString("Input: ", UNITTEST_PARAMETER_COLOR) + + HexColorString(sInput, UNITTEST_PARAMETER_INPUT); + + HandleUnitTestOutput(sInput); + } + + if (sExpected != "") + { + sExpected = _GetIndent() + HexColorString("Expected: ", UNITTEST_PARAMETER_COLOR) + + HexColorString(sExpected, UNITTEST_PARAMETER_INPUT); + + HandleUnitTestOutput(sExpected); + } + + if (sReceived != "") + { + sReceived = _GetIndent() + HexColorString("Received: ", UNITTEST_PARAMETER_COLOR) + + HexColorString(sReceived, UNITTEST_PARAMETER_RECEIVED); + + HandleUnitTestOutput(sReceived); + } + Outdent(); +} + +void DescribeTestTime(int nTime) +{ + if (nTime <= 0) + return; + + Indent(); + string sTimer = _f(nTime / 1000000.0); + string sTime = _GetIndent() + HexColorString("Test Time: ", UNITTEST_PARAMETER_COLOR) + + HexColorString(sTimer + "s", UNITTEST_PARAMETER_INPUT); + Outdent(); + + HandleUnitTestOutput(sTime); +} + +void DescribeGroupTime(int nTime) +{ + if (nTime <= 0) + return; + + string sTimer = _f(nTime / 1000000.0); + string sTime = _GetIndent() + HexColorString("Group Time: ", UNITTEST_PARAMETER_COLOR) + + HexColorString(sTimer + "s", UNITTEST_PARAMETER_INPUT); + + HandleUnitTestOutput(sTime); +} + +int Assert(string sTest, int bAssertion) +{ + sTest = HexColorString("Test ", UNITTEST_TITLE_COLOR) + + HexColorString(sTest, UNITTEST_NAME_COLOR); + + HandleUnitTestOutput(_GetIndent() + sTest + TEST_DELIMITER + (bAssertion ? TEST_PASS : TEST_FAIL)); + + if (!bAssertion) + HandleUnitTestFailure(sTest); + + return UNITTEST_ALWAYS_EXPAND ? FALSE : bAssertion; +} + +int AssertGroup(string sGroup, int bAssertion) +{ + sGroup = HexColorString("Test Group ", UNITTEST_TITLE_COLOR) + + HexColorString(sGroup, UNITTEST_NAME_COLOR); + + HandleUnitTestOutput(_GetIndent() + sGroup + TEST_DELIMITER + (bAssertion ? TEST_PASS : TEST_FAIL)); + Indent(); + + if (!bAssertion) + HandleUnitTestFailure(sGroup); + + return UNITTEST_ALWAYS_EXPAND ? FALSE : bAssertion; +} diff --git a/_module/nss/util_i_variables.nss b/_module/nss/util_i_variables.nss new file mode 100644 index 0000000..d11b405 --- /dev/null +++ b/_module/nss/util_i_variables.nss @@ -0,0 +1,2352 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_variables.nss +/// @author Ed Burke (tinygiant98) +/// @brief Functions for managing database variables. +/// ---------------------------------------------------------------------------- + +/// @details The functions in this include are meant to complement and extend +/// the game's basic variable handling functions, such as GetLocalInt() and +/// SetLocalString(). These functions allow variable storage in the module's +/// volatile sqlite database, the module's persistent campaign database, and the +/// player's sqlite database, as well as movement of variables to and from game +/// objects and various databases. Configuration options for this utility can be +/// set in `util_c_variables.nss`. +/// +/// Concepts: +/// - Databases: There are three sqlite database types available to store +/// variables: a player's bic-based db, the module's volatile db and +/// the external/persistent campaign db. When calling a function that +/// requires a database object reference (such as param oDatabase), it +/// must be a player object, DB_MODULE or DB_CAMPAIGN. All other values +/// will result in the function failing with a message to the game's log. +/// - Tag: Any Set, Increment, Decrement or Append function allows a variable +/// to be tagged with a string value of any composition or length. This +/// tag is designed to be used to group values for future delete or copy +/// operations, but may be used for any other purpose. It is important +/// to understan that the tag field is part of the primary key, which makes +/// each record unique. Although the tag is optional, if included, it must +/// be included in each subsequent call to ensure the correct variable +/// record is being operated on. +/// - Timestamp: Any Set, Increment, Decrement or Append function updates +/// the time at which the variables was set or updated. This time can be +/// be used in advanced query functions to copy or delete specific variables +/// by group. +/// - Glob/wildcard Syntax: There are several functions which allow criteria +/// to be specified to retrieve or delete variables. These criteria +/// allow the use of bitmasked types and glob syntax. If the function +/// description specified this ability, the following syntax is allowed: +/// nType - Can be a single variable type, such as +/// VARIABLE_TYPE_INT, or a bitmasked set of variable types, +/// such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. Other +/// normal bitwise operators are also allowed. To select +/// all variables types except integer, the value can be +/// passed as ~VARIABLE_TYPE_INT. Pass VARIABLE_TYPE_ALL +/// to ignore variable types. Passing VARIABLE_TYPE_NONE will +/// generally result in zero returned results. +/// sVarName - Can be an exact varname as previously set, or +/// will accept any wildcards or sets allowed by glob: +/// **Glob operations are case-senstive** +/// * - 0 or more characters +/// ? - Any single character +/// [a-j] - Any single character in the range a-j +/// [a-zA-Z] - Any single upper or lowercase letter +/// [0-9] - Any single digit +/// [^cde] - Any single character not in [cde] +/// Pass "" to ignore varnames. +/// sTag - Can be an exact tag as previously set, or will accept +/// any wildcards or sets allowed by glob. See previous +/// examples for sVarName. Pass "" to ignore tags. +/// nTime - Filter results by timestamp. A timestamp is set on +/// the variable anytime a variable is inserted or updated. +/// If nTime is negative, the system will match all variables +/// set before nTime. If nTime is positive, the system will +/// match all variables set after nTime. The easiest way to +/// understand this concept is to determine the time you want +/// to compare against (in Unix seconds), then pass that time +/// as negative to seek variables set/updated before that time, +/// or positive to seek variables set/updated after that time. +/// Pass 0 to ignore timestamps. +/// +/// Advanced Usage: +/// - Copying from Database to Locals: `CopyDatabaseVariablesToObject()` +/// allows specified database variables to any valid game object. +/// Local variables do not allow additional fields that are retrieved +/// from the database, so the function `DatabaseToObjectVarName()` is +/// provided in `util_c_variables.nss` to allow users to construct +/// unique varnames for a copied database variable. See glob/wildcard +/// syntax concept above for how to use parameters in this function. +/// +/// - Copying from Locals to Database: `CopyLocalVariablesToDatabase()` +/// allows specified local variables from any game object (except the +/// module object) to any database. + + + +/// - Copying from Locals to Database: There are three functions which allow +/// variables which meet specific criteria to be copied from a game object +/// to a specified database. Local variables do not have tags, however, a +/// tag can be supplied to these functions and the tag will be saved into +/// the database. These methods may be useful to save current object +/// state into a persistent database to be later retrieved individually +/// of by mass copy with a database -> local copy method. +/// +/// - Record uniqueness: Module, Player and Persistent variables are stored +/// in sqlite databases. Each record is unique based on variable type, +/// name and tag. The variable tag is optional. This behavior allows +/// multiple variables with the same type and name, but with different +/// tags. If using tags, it is incumbent upon the user to include the +/// desired tag is in all functions calls to ensure the correct record +/// is operated on. + +#include "util_i_debug" +#include "util_i_lists" +#include "util_i_matching" +#include "util_c_variables" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +const int VARIABLE_TYPE_NONE = 0x00; +const int VARIABLE_TYPE_INT = 0x01; +const int VARIABLE_TYPE_FLOAT = 0x02; +const int VARIABLE_TYPE_STRING = 0x04; +const int VARIABLE_TYPE_OBJECT = 0x08; +const int VARIABLE_TYPE_VECTOR = 0x10; +const int VARIABLE_TYPE_LOCATION = 0x20; +const int VARIABLE_TYPE_JSON = 0x40; +const int VARIABLE_TYPE_SERIALIZED = 0x80; +const int VARIABLE_TYPE_ALL = 0xff; + +const string VARIABLE_OBJECT = "VARIABLE:OBJECT"; +const string VARIABLE_CAMPAIGN = "VARIABLE:CAMPAIGN"; + +object DB_MODULE = GetModule(); +object DB_CAMPAIGN = OBJECT_INVALID; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Creates a variable table in oObject's database. +/// @param oObject Optional object reference. If passed, should +/// be a PC object or a db object (DB_MODULE || DB_CAMPAIGN). +/// @note This function is never required to be called separately +/// during OnModuleLoad. Table creation is handled during +/// the variable setting process. +void CreateVariableTable(object oObject); + +// ----------------------------------------------------------------------------- +// Local Variables +// ----------------------------------------------------------------------------- + +/// @brief Returns a json array of all local variables on oObject. +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param nType VARIABLE_TYPE_* constant for type of variable to retrieve. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to retrieve. Accepts glob wildcard and +/// set syntax. +/// @returns a JSON array of variables set on oObject. The array will consist +/// of JSON objects with the following key:value pairs: +/// type: {int} Reference to VARIABLE_TYPE_* +/// value: {type} Type depends on type +/// -- objects will be returned as a string object id which +/// can be used in StringToObject() +/// varname: {string} +json GetLocalVariables(object oObject, int nType = VARIABLE_TYPE_ALL, string sVarName = "*"); + +/// @brief Deletes local variables from oObject. +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param nType VARIABLE_TYPE_* constant for type of variable to delete. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to delete. Accepts glob wildcard and +/// set syntax. +void DeleteLocalVariables(object oObject, int nType = VARIABLE_TYPE_NONE, string sVarName = ""); + +/// @brief Copies local variables from oObject to another game object oTarget. +/// @param oSource Game object to get local variables from. This method will +/// not work on the module object. +/// @param oTarget The game object to copy local variables to. +/// @param nType VARIABLE_TYPE_* constant for type of variable to copy. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to copy. Accepts glob wildcard and +/// set syntax. +/// @param bDelete If TRUE, deletes the local variables from oSource after they +/// are copied oTarget. +/// @note This method *can* be used to set variables onto the module object. +void CopyLocalVariablesToObject(object oSource, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", int bDelete = TRUE); + +/// @brief Copies local variables from oSource to oDatabase. +/// @param oSource Game object to get local variables from. This method will +/// not work on the module object. +/// @param oDatabase Database to copy variables to (PC Object || DB_MODULE || DB_CAMPAIGN). +/// @param nType VARIABLE_TYPE_* constant for type of variable to copy. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to copy. Accepts glob wildcard and +/// set syntax. +/// @param sTag Optional tag reference. All variables copied with this function +/// will have sTag applied. +/// @param bDelete If TRUE, deletes the local variables from oSource after they +/// are copied to the module database. +void CopyLocalVariablesToDatabase(object oSource, object oDatabase, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int bDelete = TRUE); + +/// @brief Copies variables from an sqlite database to a game object as local variables. +/// @param oDatabase Database to copy variables to (PC Object || DB_MODULE || DB_CAMPAIGN). +/// @param oTarget Game object to set local variables on. +/// @param nType VARIABLE_TYPE_* constant for type of variable to copy. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to copy. Accepts glob wildcard and +/// set syntax. +/// @param sTag Optional tag reference. Accepts glob wildcard and set syntax. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @param bDelete If TRUE, deletes the local variables from oObject after they +/// are copied to the module database. +/// @note This method *can* be used to set variables onto the module object. +void CopyDatabaseVariablesToObject(object oDatabase, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0, int bDelete = TRUE); + +/// @brief Copies variables from an sqlite database to another sqlite database. +/// @param oSource Database to copy variables from (PC Object || DB_MODULE || DB_CAMPAIGN). +/// @param oTarget Database to copy variables to (PC Object || DB_MODULE || DB_CAMPAIGN). +/// @param nType VARIABLE_TYPE_* constant for type of variable to copy. +/// Accepts bitmasked types such as VARIABLE_TYPE_INT | VARIABLE_TYPE_FLOAT. +/// @param sVarName Name of variable to copy. Accepts glob wildcard and +/// set syntax. +/// @param sTag Optional tag reference. Accepts glob wildcard and set syntax. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @param bDelete If TRUE, deletes the local variables from oObject after they +/// are copied to the module database. +void CopyDatabaseVariablesToDatabase(object oSource, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0, int bDelete = TRUE); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalInt(object oObject, string sVarName); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalFloat(object oObject, string sVarName); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalString(object oObject, string sVarName); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalObject(object oObject, string sVarName); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalLocation(object oObject, string sVarName); + +/// @brief Determines whether a local variable has been set on oObject +/// @param oObject Game object to get local variables from. This method will +/// not work on the module object. +/// @param sVarName Name of variable to retrieve. This must be the exact varname, +/// glob wildcards and sets are not accepted. +int HasLocalJson(object oObject, string sVarName); + +// ----------------------------------------------------------------------------- +// Module Database +// ----------------------------------------------------------------------------- + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param nValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleInt(string sVarName, int nValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param fValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleFloat(string sVarName, float fValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleString(string sVarName, string sValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleObject(string sVarName, object oValue, string sTag = ""); + +/// @brief Set a serialized object into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +/// @note This function will serialize the passed object. To store an object by +/// reference, use SetModuleObject(). +void SetModuleSerialized(string sVarName, object oValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param lValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleLocation(string sVarName, location lValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param vValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleVector(string sVarName, vector vValue, string sTag = ""); + +/// @brief Set a variable into the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param jValue Value of the variable. +/// @param sTag Optional tag reference. +void SetModuleJson(string sVarName, json jValue, string sTag = ""); + +/// @brief Set a previously set variable's tag to sTag. +/// @param nType VARIABLE_TYPE_* constant. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +/// @param sNewTag New tag to assign. +void SetModuleVariableTag(int nType, string sVarName, string sTag = "", string sNewTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0. +/// @param sTag Optional tag reference. +int GetModuleInt(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0.0. +/// @param sTag Optional tag reference. +float GetModuleFloat(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise "". +/// @param sTag Optional tag reference. +string GetModuleString(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise OBJECT_INVALID. +/// @param sTag Optional tag reference. +object GetModuleObject(string sVarName, string sTag = ""); + +/// @brief Retrieve and create a serialized object from the module's +/// volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Tag reference. +/// @param l Location to create the deserialized object. +/// @param oTarget Target object on which to create the deserialized object. +/// @returns The requested serialized object, if found, otherwise +/// OBJECT_INVALID. +/// @note If oTarget is passed and has inventory, the retrieved object +/// will be created in oTarget's inventory, otherwise it will be created +/// at location l. +object GetModuleSerialized(string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise LOCATION_INVALID. +/// @param sTag Optional tag reference. +location GetModuleLocation(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise Vector(). +/// @param sTag Optional tag reference. +vector GetModuleVector(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise JsonNull(). +/// @param sTag Optional tag reference. +json GetModuleJson(string sVarName, string sTag = ""); + +/// @brief Returns a json array of key-value pairs. +/// @param nType VARIABLE_TYPE_*, accepts bitmasked values. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, all variables will be returned. +/// @details This function will return an array of json objects containing +/// information about each variable found. Each json object in the +/// array will contain the following key-value pairs: +/// tag: {string} +/// timestamp: {int} UNIX seconds +/// type: {int} Reference to VARIABLE_TYPE_* +/// value: {type} Type depends on type +/// -- objects will be returned as a string object id which +/// can be used in StringToObject() +/// -- serialized objects will be returned as their json +/// representation and can be used in JsonToObject() +/// varname: {string} +json GetModuleVariablesByPattern(int nType = VARIABLE_TYPE_ALL, string sVarName = "", + string sTag = "", int nTime = 0); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +int DeleteModuleInt(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +float DeleteModuleFloat(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +string DeleteModuleString(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +object DeleteModuleObject(string sVarName, string sTag = ""); + +/// @brief Delete a serialized object from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +void DeleteModuleSerialized(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +location DeleteModuleLocation(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +vector DeleteModuleVector(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the module's volatile sqlite database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +json DeleteModuleJson(string sVarName, string sTag = ""); + +/// @brief Deletes all variables from the module's volatile sqlite database. +/// @warning Calling this method will result in all variables in the module's +/// volatile sqlite database being deleted without additional warning. +void DeleteModuleVariables(); + +/// @brief Deletes all variables from the module's volatile sqlite database +/// that match the parameter criteria. +/// @param nType Bitwise VARIABLE_TYPE_*. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, no variables will be returned. +/// @warning Calling this method without passing any parameters will result +/// in all variables in the module's volatile sqlite database being +/// deleted without additional warning. +void DeleteModuleVariablesByPattern(int nType = VARIABLE_TYPE_NONE, string sVarName = "", + string sTag = "*", int nTime = 0); + +/// @brief Increments an integer variable in the module's volatile sqlite +/// database by nIncrement. If the variable doesn't exist, it will be +/// initialized to 0 before incrementing. +/// @param sVarName Name of the variable. +/// @param nIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positive, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +int IncrementModuleInt(string sVarName, int nIncrement = 1, string sTag = ""); + +/// @brief Decrements an integer variable in the module's volatile sqlite +/// database by nDecrement. If the variable doesn't exist, it will be +/// initialized to 0 before decrementing. +/// @param sVarName Name of the variable. +/// @param nDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +int DecrementModuleInt(string sVarName, int nDecrement = -1, string sTag = ""); + +/// @brief Increments an float variable in the module's volatile sqlite +/// database by nIncrement. If the variable doesn't exist, it will be +/// initialized to 0.0 before incrementing. +/// @param sVarName Name of the variable. +/// @param fIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positing, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +float IncrementModuleFloat(string sVarName, float fIncrement = 1.0, string sTag = ""); + +/// @brief Decrements an float variable in the module's volatile sqlite +/// database by nDecrement. If the variable doesn't exist, it will be +/// initialized to 0.0 before decrementing. +/// @param sVarName Name of the variable. +/// @param fDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is a positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +float DecrementModuleFloat(string sVarName, float fDecrement = -1.0, string sTag = ""); + +/// @brief Appends sAppend to the end of a string variable in the module's +/// volatile sqlite database. If the variable doesn't exist, it will be +/// initialized to "" before appending. +/// @param sVarName Name of the variable. +/// @param sAppend Value to append. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after appending. +string AppendModuleString(string sVarName, string sAppend, string sTag = ""); + +// ----------------------------------------------------------------------------- +// Player Database +// ----------------------------------------------------------------------------- + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param nValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerInt(object oPlayer, string sVarName, int nValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param fValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerFloat(object oPlayer, string sVarName, float fValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerString(object oPlayer, string sVarName, string sValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerObject(object oPlayer, string sVarName, object oValue, string sTag = ""); + +/// @brief Set a serialized object into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +/// @note This function will serialize the passed object. To store an object by +/// reference, use SetPlayerObject(). +void SetPlayerSerialized(object oPlayer, string sVarName, object oValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param lValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerLocation(object oPlayer, string sVarName, location lValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param vValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerVector(object oPlayer, string sVarName, vector vValue, string sTag = ""); + +/// @brief Set a variable into the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param jValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPlayerJson(object oPlayer, string sVarName, json jValue, string sTag = ""); + +/// @brief Set a previously set variable's tag to sTag. +/// @param oPlayer Player object reference. +/// @param nType VARIABLE_TYPE_* constant. +/// @param sVarName Name of the variable. +/// @param sTag Tag reference. +void SetPlayerVariableTag(object oPlayer, int nType, string sVarName, string sTag = "", string sNewTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0. +/// @param sTag Optional tag reference. +int GetPlayerInt(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0.0. +/// @param sTag Optional tag reference. +float GetPlayerFloat(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise "". +/// @param sTag Optional tag reference. +string GetPlayerString(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise OBJECT_INVALID. +/// @param sTag Optional tag reference. +object GetPlayerObject(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve and create a serialized object from the player's sqlite +/// database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param l Location to create the deserialized object. +/// @param oTarget Target object on which to create the deserialized object. +/// @returns The requested serialized object, if found, otherwise +/// OBJECT_INVALID. +/// @note If oTarget is passed and has inventory, the retrieved object +/// will be created in oTarget's inventory, otherwise it will be created +/// at location l. +object GetPlayerSerialized(object oPlayer, string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise LOCATION_INVALID. +/// @param sTag Optional tag reference. +location GetPlayerLocation(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise Vector(). +/// @param sTag Optional tag reference. +vector GetPlayerVector(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise JsonNull(). +/// @param sTag Optional tag reference. +json GetPlayerJson(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Returns a json array of key-value pairs. +/// @param oPlayer Player object reference. +/// @param nType VARIABLE_TYPE_*, accepts bitmasked values. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, all variables will be returned. +/// @details This function will return an array of json objects containing +/// information about each variable found. Each json object in the +/// array will contain the following key-value pairs: +/// tag: {string} +/// timestamp: {int} UNIX seconds +/// type: {int} Reference to VARIABLE_TYPE_* +/// value: {type} Type depends on type +/// -- objects will be returned as a string object id which +/// can be used in StringToObject() +/// -- serialized objects will be returned as their json +/// representation and can be used in JsonToObject() +/// varname: {string} +json GetPlayerVariablesByPattern(object oPlayer, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +int DeletePlayerInt(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +float DeletePlayerFloat(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +string DeletePlayerString(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +object DeletePlayerObject(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a serialized object from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +void DeletePlayerSerialized(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +location DeletePlayerLocation(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +vector DeletePlayerVector(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Delete a variable from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +json DeletePlayerJson(object oPlayer, string sVarName, string sTag = ""); + +/// @brief Deletes all variables from the player's sqlite database. +/// @param oPlayer Player object reference. +/// @warning Calling this method will result in all variables in the module's +/// volatile sqlite database being deleted without additional warning. +void DeletePlayerVariables(object oPlayer); + +/// @brief Deletes all variables from the player's sqlite database +/// that match the parameter criteria. +/// @param oPlayer Player object reference. +/// @param nType Bitwise VARIABLE_TYPE_*. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, no variables will be returned. +/// @warning Calling this method without passing any parameters will result +/// in all variables in the player's sqlite database being +/// deleted without additional warning. +void DeletePlayerVariablesByPattern(object oPlayer, int nType = VARIABLE_TYPE_NONE, + string sVarName = "", string sTag = "", int nTime = 0); + +/// @brief Increments an integer variable in the player's sqlite database. +/// If the variable doesn't exist, it will be initialized to 0 before +/// incrementing. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param nIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positive, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +int IncrementPlayerInt(object oPlayer, string sVarName, int nIncrement = 1, string sTag = ""); + +/// @brief Decrements an integer variable in the player's sqlite database. +/// If the variable doesn't exist, it will be initialized to 0 before +/// decrementing. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param nDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +int DecrementPlayerInt(object oPlayer, string sVarName, int nDecrement = -1, string sTag = ""); + +/// @brief Increments an float variable in the player's sqlite database. +/// If the variable doesn't exist, it will be initialized to 0.0 before +/// incrementing. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param fIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positing, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +float IncrementPlayerFloat(object oPlayer, string sVarName, float fIncrement = 1.0, string sTag = ""); + +/// @brief Decrements an float variable in the player's sqlite database. +/// If the variable doesn't exist, it will be initialized to 0.0 before +/// decrementing. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param fDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is a positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +float DecrementPlayerFloat(object oPlayer, string sVarName, float fDecrement = -1.0, string sTag = ""); + +/// @brief Appends sAppend to the end of a string variable in the player's +/// sqlite database. If the variable doesn't exist, it will be +/// initialized to "" before appending. +/// @param oPlayer Player object reference. +/// @param sVarName Name of the variable. +/// @param sAppend Value to append. +/// @param sTag Optional tag reference. +/// @returns The value of the variable after appending. +string AppendPlayerString(object oPlayer, string sVarName, string sAppend, string sTag = ""); + +// ----------------------------------------------------------------------------- +// Campaign Database +// ----------------------------------------------------------------------------- + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param nValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentInt(string sVarName, int nValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param fValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentFloat(string sVarName, float fValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param sValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentString(string sVarName, string sValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentObject(string sVarName, object oValue, string sTag = ""); + +/// @brief Set a serialized object into the campaign database. +/// @param sVarName Name of the variable. +/// @param oValue Value of the variable. +/// @param sTag Optional tag reference. +/// @note This function will serialize the passed object. To store an object by +/// reference, use SetPersistentObject(). +void SetPersistentSerialized(string sVarName, object oValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param lValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentLocation(string sVarName, location lValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param vValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentVector(string sVarName, vector vValue, string sTag = ""); + +/// @brief Set a variable into the campaign database. +/// @param sVarName Name of the variable. +/// @param jValue Value of the variable. +/// @param sTag Optional tag reference. +void SetPersistentJson(string sVarName, json jValue, string sTag = ""); + +/// @brief Set a previously set variable's tag to sTag. +/// @param nType VARIABLE_TYPE_* constant. +/// @param sVarName Name of the variable. +/// @param sTag Tag reference. +void SetPersistentVariableTag(int nType, string sVarName, string sTag = "", string sNewTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0. +/// @param sTag Optional tag reference. +int GetPersistentInt(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise 0.0. +/// @param sTag Optional tag reference. +float GetPersistentFloat(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise "". +/// @param sTag Optional tag reference. +string GetPersistentString(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise OBJECT_INVALID. +/// @param sTag Optional tag reference. +object GetPersistentObject(string sVarName, string sTag = ""); + +/// @brief Retrieve and create a serialized object from the campaign database. +/// @param sVarName Name of the variable. +/// @param l Location to create the deserialized object. +/// @param oTarget Target object on which to create the deserialized object. +/// @returns The requested serialized object, if found, otherwise +/// OBJECT_INVALID. +/// @note If oTarget is passed and has inventory, the retrieved object +/// will be created in oTarget's inventory, otherwise it will be created +/// at location l. +object GetPersistentSerialized(string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise LOCATION_INVALID. +/// @param sTag Optional tag reference. +location GetPersistentLocation(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise Vector(). +/// @param sTag Optional tag reference. +vector GetPersistentVector(string sVarName, string sTag = ""); + +/// @brief Retrieve a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @returns Variable value, if found, otherwise JsonNull(). +/// @param sTag Optional tag reference. +json GetPersistentJson(string sVarName, string sTag = ""); + +/// @brief Returns a json array of key-value pairs. +/// @param nType VARIABLE_TYPE_*, accepts bitmasked values. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, all variables will be returned. +/// @details This function will return an array of json objects containing +/// information about each variable found. Each json object in the +/// array will contain the following key-value pairs: +/// tag: {string} +/// timestamp: {int} UNIX seconds +/// type: {int} Reference to VARIABLE_TYPE_* +/// value: {type} Type depends on type +/// -- objects will be returned as a string object id which +/// can be used in StringToObject() +/// -- serialized objects will be returned as their json +/// representation and can be used in JsonToObject() +/// varname: {string} +json GetPersistentVariablesByPattern(int nType = VARIABLE_TYPE_ALL, string sVarName = "*", + string sTag = "*", int nTime = 0); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +int DeletePersistentInt(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +float DeletePersistentFloat(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +string DeletePersistentString(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +object DeletePersistentObject(string sVarName, string sTag = ""); + +/// @brief Delete a serialized object from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +void DeletePersistentSerialized(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +location DeletePersistentLocation(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +vector DeletePersistentVector(string sVarName, string sTag = ""); + +/// @brief Delete a variable from the campaign database. +/// @param sVarName Name of the variable. +/// @param sTag Optional tag reference. +json DeletePersistentJson(string sVarName, string sTag = ""); + +/// @brief Deletes all variables from the campaign database. +/// @warning Calling this method will result in all variables in the campaign +/// database being deleted without additional warning. +void DeletePersistentVariables(); + +/// @brief Deletes all variables from the campaign database that match the +/// parameter criteria. +/// @param nType Bitwise VARIABLE_TYPE_*. +/// @param sVarName Variable name pattern, accepts glob patterns, sets +/// and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @note If no parameters are passed, no variables will be deleted. +/// @warning Calling this method without passing any parameters will result +/// in all variables in the campaign database being +/// deleted without additional warning. +void DeletePersistentVariablesByPattern(int nType = VARIABLE_TYPE_NONE, string sVarName = "", + string sTag = "", int nTime = 0); + +/// @brief Increments an integer variable in the campaign database by nIncrement. +/// If the variable doesn't exist, it will be initialized to 0 before +/// incrementing. +/// @param sVarName Name of the variable. +/// @param nIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positive, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +int IncrementPersistentInt(string sVarName, int nIncrement = 1, string sTag = ""); + +/// @brief Decrements an integer variable in the campaign database by nDecrement. +/// If the variable doesn't exist, it will be initialized to 0 before +/// decrementing. +/// @param sVarName Name of the variable. +/// @param nDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +int DecrementPersistentInt(string sVarName, int nDecrement = -1, string sTag = ""); + +/// @brief Increments an float variable in the campaign database by nIncrement. +/// If the variable doesn't exist, it will be initialized to 0.0 before +/// incrementing. +/// @param sVarName Name of the variable. +/// @param fIncrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after incrementing. +/// @note nIncrement is expected to be positing, however, this method will +/// accept a negative value for nIncrement and will decrement the variable +/// value. +float IncrementPersistentFloat(string sVarName, float fIncrement = 1.0, string sTag = ""); + +/// @brief Decrements an float variable in the campaign database by nDecrement. +/// If the variable doesn't exist, it will be initialized to 0.0 before +/// decrementing. +/// @param sVarName Name of the variable. +/// @param fDecrement Amount to increment the variable by. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after decrementing. +/// @note nDecrement is expected to be negative. If nDecrement is a positive, +/// this method will decrement the variable by nDecrement and will not +/// fallback to incrementing behavior. +float DecrementPersistentFloat(string sVarName, float fDecrement = -1.0, string sTag = ""); + +/// @brief Appends sAppend to the end of a string variable in the campaign +/// database. If the variable doesn't exist, it will be initialized to "" +/// before appending. +/// @param sVarName Name of the variable. +/// @param sAppend Value to append. +/// @param sTag Optional tag reference. Only used if the variable was not +/// previously set. +/// @returns The value of the variable after appending. +string AppendPersistentString(string sVarName, string sAppend, string sTag = ""); + +// ----------------------------------------------------------------------------- +// Private Functions +// ----------------------------------------------------------------------------- + +/// @private Returns the variable type as a string +/// @note For debug purposes only. +string _VariableTypeToString(int nType) +{ + if (nType == VARIABLE_TYPE_INT) return "INT"; + else if (nType == VARIABLE_TYPE_FLOAT) return "FLOAT"; + else if (nType == VARIABLE_TYPE_STRING) return "STRING"; + else if (nType == VARIABLE_TYPE_OBJECT) return "OBJECT"; + else if (nType == VARIABLE_TYPE_VECTOR) return "VECTOR"; + else if (nType == VARIABLE_TYPE_LOCATION) return "LOCATION"; + else if (nType == VARIABLE_TYPE_JSON) return "JSON"; + else if (nType == VARIABLE_TYPE_SERIALIZED) return "SERIALIZED"; + else if (nType == VARIABLE_TYPE_NONE) return "NONE"; + else if (nType == VARIABLE_TYPE_ALL) return "ALL"; + else return "UNKNOWN"; +} + +/// @private Converts an NWN type to a VARIABLE_TYPE_* +int _TypeToVariableType(json jType) +{ + int nType = JsonGetInt(jType); + + if (nType == 1) return VARIABLE_TYPE_INT; + else if (nType == 2) return VARIABLE_TYPE_FLOAT; + else if (nType == 3) return VARIABLE_TYPE_STRING; + else if (nType == 4) return VARIABLE_TYPE_OBJECT; + else if (nType == 5) return VARIABLE_TYPE_LOCATION; + else if (nType == 7) return VARIABLE_TYPE_JSON; + return VARIABLE_TYPE_NONE; +} + +/// @private Converts VARIABLE_TYPE_* bitmask to IN +string _VariableTypeToArray(int nTypes) +{ + if (nTypes == VARIABLE_TYPE_NONE) return ""; + else if (nTypes == VARIABLE_TYPE_ALL) return "1,2,3,4,5,6,7"; + + string sArray; + if (nTypes & VARIABLE_TYPE_INT) sArray = AddListItem(sArray, "1"); + if (nTypes & VARIABLE_TYPE_FLOAT) sArray = AddListItem(sArray, "2"); + if (nTypes & VARIABLE_TYPE_STRING) sArray = AddListItem(sArray, "3"); + if (nTypes & VARIABLE_TYPE_OBJECT) sArray = AddListItem(sArray, "4"); + if (nTypes & VARIABLE_TYPE_LOCATION) sArray = AddListItem(sArray, "5"); + if (nTypes & VARIABLE_TYPE_JSON) sArray = AddListItem(sArray, "7"); + + return sArray; +} + +/// @private Deletes a single local variable +void _DeleteLocalVariable(object oObject, string sVarName, int nType) +{ + if (nType == VARIABLE_TYPE_INT) DeleteLocalInt(oObject, sVarName); + else if (nType == VARIABLE_TYPE_FLOAT) DeleteLocalFloat(oObject, sVarName); + else if (nType == VARIABLE_TYPE_STRING) DeleteLocalString(oObject, sVarName); + else if (nType == VARIABLE_TYPE_OBJECT) DeleteLocalObject(oObject, sVarName); + else if (nType == VARIABLE_TYPE_LOCATION) DeleteLocalLocation(oObject, sVarName); + else if (nType == VARIABLE_TYPE_JSON) DeleteLocalJson(oObject, sVarName); +} + +/// @private Sets a single local variable +void _SetLocalVariable(object oObject, string sVarName, int nType, json jValue) +{ + if (nType == VARIABLE_TYPE_INT) SetLocalInt(oObject, sVarName, JsonGetInt(jValue)); + else if (nType == VARIABLE_TYPE_FLOAT) SetLocalFloat(oObject, sVarName, JsonGetFloat(jValue)); + else if (nType == VARIABLE_TYPE_STRING) SetLocalString(oObject, sVarName, JsonGetString(jValue)); + else if (nType == VARIABLE_TYPE_OBJECT) SetLocalObject(oObject, sVarName, StringToObject(JsonGetString(jValue))); + else if (nType == VARIABLE_TYPE_LOCATION) SetLocalLocation(oObject, sVarName, JsonToLocation(jValue)); + else if (nType == VARIABLE_TYPE_JSON) SetLocalJson(oObject, sVarName, jValue); +} + +/// @private Prepares an query against an object (module/player). Ensures +/// appropriate tables have been created before attempting query. +sqlquery _PrepareQueryObject(object oObject, string sQuery) +{ + CreateVariableTable(oObject); + return SqlPrepareQueryObject(oObject, sQuery); +} + +/// @private Prepares an query against an campaign database. Ensures +/// appropriate tables have been created before attempting query. +sqlquery _PrepareQueryCampaign(string sQuery) +{ + CreateVariableTable(DB_CAMPAIGN); + return SqlPrepareQueryCampaign(VARIABLE_CAMPAIGN_DATABASE, sQuery); +} + +/// @private Prepares a select query to retrieve a variable value stored +/// in any database. +sqlquery _PrepareVariableSelect(object oObject, int nType, string sVarName, string sTag) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "SELECT value FROM " + sTable + " WHERE type = @type " + + "AND varname GLOB @varname AND tag GLOB @tag;"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindInt (q, "@type", nType); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag", sTag); + return q; +} + +/// @private Prepares an insert query to stored a variable in any database. +sqlquery _PrepareVariableInsert(object oObject, int nType, string sVarName, string sTag) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "INSERT INTO " + sTable + " (type, varname, value, tag, timestamp) " + + "VALUES (@type, @varname, IIF(json_valid(@value), @value ->> '$', @value), " + + "@tag, strftime('%s', 'now')) ON CONFLICT (type, varname, tag) DO UPDATE " + + "SET value = IIF(json_valid(@value), @value ->> '$', @value), tag = @tag, " + + "timestamp = strftime('%s', 'now');"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindInt (q, "@type", nType); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag", sTag); + return q; +} + +/// @private Prepares an update query to modify the tag assicated with a variable. +sqlquery _PrepareTagUpdate(object oObject, int nType, string sVarName, string sTag1, string sTag2) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "UPDATE " + sTable + " SET tag = @tag2 WHERE type = @type " + + "AND varname GLOB @varname AND tag GLOB tag1;"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindInt (q, "@type", nType); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag1", sTag1); + SqlBindString(q, "@tag2", sTag2); + return q; +} + +/// @private Prepares an delete query to remove a variable stored in any database. +sqlquery _PrepareSimpleVariableDelete(object oObject, int nType, string sVarName, string sTag) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "DELETE FROM " + sTable + " WHERE type = @type " + + "AND varname GLOB @varname AND tag GLOB @tag " + + "RETURNING value;"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindInt (q, "@type", nType); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag", sTag); + return q; +} + +/// @private Prepares a complex delete query to remove multiple variables by criteria. +/// @param nType Bitwise VARIABLE_TYPE_*. +/// @param sVarName Variable name pattern, accept glob patterns, sets and wildcards. +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards. +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +sqlquery _PrepareComplexVariableDelete(object oObject, int nType, string sVarName, string sTag, int nTime) +{ + int n, bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string sWhere = (sVarName == "" ? "" : " $" + IntToString(++n) + " varname GLOB @varname"); + sWhere += (sTag == "" ? "" : " $" + IntToString(++n) + " tag GLOB @tag"); + sWhere += (nType <= 0 ? "" : " $" + IntToString(++n) + " (type & @type) > 0"); + sWhere += (nTime == 0 ? "" : " $" + IntToString(++n) + " timestamp " + (nTime > 0 ? ">" : "<") + " @time"); + + json jKeyWords = ListToJson("WHERE,AND,AND,AND"); + string s = SubstituteString("DELETE FROM " + sTable + sWhere + ";", jKeyWords); + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + if (sVarName != "") SqlBindString(q, "@varname", sVarName); + if (sTag != "") SqlBindString(q, "@tag", sTag); + if (nType > 0) SqlBindInt (q, "@type", nType); + if (nTime != 0) SqlBindInt (q, "@time", abs(nTime)); + return q; +} + +/// @private Retrieves variables from database associated with oObject and returns +/// selected variables in a json array containing variable metadata and value. +/// @param nType Bitwise VARIABLE_TYPE_* +/// @param sVarName Variable name pattern, accept glob patterns, sets and wildcards +/// @param sTag Tag pattern, accepts glob patterns, sets and wildcards +/// @param nTime A positive value will filter for timestamps after +/// nTime, a negative value will filter for timestamps before nTime. +/// @warning If no parameters are passed, this query will result in no variables being +/// retrieved. +json _DatabaseVariablesToJson(object oObject, int nType, string sVarName, string sTag, int nTime) +{ + int n, bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string sWhere = (sVarName == "" ? "" : " $" + IntToString(++n) + " varname GLOB @varname"); + sWhere += (sTag == "" ? "" : " $" + IntToString(++n) + " tag GLOB @tag"); + sWhere += (nType <= 0 ? "" : " $" + IntToString(++n) + " (type & @type) > 0"); + sWhere += (nTime == 0 ? "" : " $" + IntToString(++n) + " timestamp " + (nTime > 0 ? ">" : "<") + " @time"); + + json jKeyWords = ListToJson("WHERE,AND,AND,AND"); + string s = "WITH json_variables AS (SELECT json_object('type', type, 'varname', varname, " + + "'tag', tag, 'value', value, 'timestamp', timestamp) AS variable_object " + + "FROM " + sTable + sWhere + ") " + + "SELECT json_group_array(json(variable_object)) FROM json_variables;"; + s = SubstituteString(s, jKeyWords); + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + if (sVarName != "") SqlBindString(q, "@varname", sVarName); + if (sTag != "") SqlBindString(q, "@tag", sTag); + if (nType > 0) SqlBindInt (q, "@type", nType); + if (nTime != 0) SqlBindInt (q, "@time", abs(nTime)); + + return SqlStep(q) ? SqlGetJson(q, 0) : JsonArray(); +} + +json _LocalVariablesToJson(object oObject, int nType, string sVarName) +{ + if (!GetIsObjectValid(oObject) || oObject == DB_MODULE) + return JsonArray(); + + json jVarTable = JsonPointer(ObjectToJson(oObject, TRUE), "/VarTable/value"); + if (!JsonGetLength(jVarTable)) + return JsonArray(); + + int n; + string sWhere = (sVarName == "" ? "" : " $" + IntToString(++n) + " variable_object ->> 'varname' GLOB @varname"); + sWhere += (nType <= 0 ? "" : " $" + IntToString(++n) + " variable_object ->> 'type' IN (" + _VariableTypeToArray(nType) + ")"); + + json jKeyWords = ListToJson("WHERE,AND"); + string s = "WITH local_variables AS (SELECT json_object('type', v.value -> 'Type.value', " + + "'varname', v.value -> 'Name.value', 'value', v.value -> 'Value.value') " + + "as variable_object FROM json_each(@vartable) as v) " + + "SELECT json_group_array(json(variable_object)) FROM local_variables " + sWhere + ";"; + s = SubstituteString(s, jKeyWords); + + sqlquery q = SqlPrepareQueryObject(DB_MODULE, s); + SqlBindJson(q, "@vartable", jVarTable); + SqlBindString(q, "@varname", sVarName); + + return SqlStep(q) ? SqlGetJson(q, 0) : JsonArray(); +} + +/// @private Increments/Decremenst an existing variable (int/float). If the variable +/// does not exist, creates variable, then increments/decrements. +sqlquery _PrepareVariableIncrement(object oObject, int nType, string sVarName, string sTag) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "INSERT INTO " + sTable + " (type, varname, value, tag, timestamp) " + + "VALUES (@type, @varname, @value, @tag, strftime('%s','now')) " + + "ON CONFLICT (type, varname, tag) DO UPDATE SET value = value + @value, " + + "timestamp = strftime('%s','now') RETURNING value;"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindInt (q, "@type", nType); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag", sTag); + return q; +} + +/// @private Appends a string to an existing variable. If the variables does not +/// exist, creates the variable, then appends. +sqlquery _PrepareVariableAppend(object oObject, string sVarName, string sTag) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "INSERT INTO " + sTable + " " + + "(type, varname, value, tag, timestamp) " + + "VALUES (@type, @varname, @value, @tag, strftime('%s', 'now')) " + + "ON CONFLICT (type, varname, tag) " + + "DO UPDATE SET value = value || @value, " + + "timestamp = strftime('%s', 'now') " + + "RETURNING value;"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlBindString(q, "@varname", sVarName); + SqlBindString(q, "@tag", sTag); + return q; +} + +/// @private Opens an sqlite transaction +void _BeginSQLTransaction(object oObject) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + + string s = "BEGIN TRANSACTION;"; + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlStep(q); +} + +/// @private Commits an open sqlite transaction +void _CommitSQLTransaction(object oObject) +{ + int bPC = GetIsPC(oObject); + int bCampaign = oObject == DB_CAMPAIGN; + + string s = "COMMIT TRANSACTION;"; + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oObject, s) : _PrepareQueryCampaign(s); + SqlStep(q); +} + +/// @private Copies specified variables from oSource (game object) to oTarget (db). +void _CopyVariablesToDatabase(object oSource, object oDatabase, int nTypes, + string sVarNames, string sTag, int bDelete) +{ + if (oSource == GetModule()) + return; + + if (!GetIsPC(oDatabase) && oDatabase != DB_MODULE && oDatabase != DB_CAMPAIGN) + { + if (IsDebugging(DEBUG_LEVEL_NOTICE)) + Notice("Attempt to copy local variables to database failed:" + + "\n oSource -> " + GetName(oSource) + + "\n oDatabase -> " + GetName(oDatabase) + + "\n nTypes -> " + IntToHexString(nTypes) + + "\n sVarName -> " + sVarNames + + "\n sTag -> " + sTag + + "\n bDelete -> " + (bDelete ? "TRUE" : "FALSE")); + return; + } + + json jVariables = GetLocalVariables(oSource, nTypes, sVarNames); + int nCount = JsonGetLength(jVariables); + + if (!nCount) + return; + + _BeginSQLTransaction(oDatabase); + int n; for (n; n < nCount; n++) + { + json jVariable = JsonPointer(jVariables, "/" + IntToString(n)); + int nType = JsonGetInt(JsonPointer(jVariable, "/type")); + string sVarName = JsonGetString(JsonPointer(jVariable, "/varname")); + json jValue = JsonPointer(jVariable, "/value"); + + sVarName = ObjectToDatabaseVarName(oSource, oDatabase, sVarName, nType, sTag); + sTag = ObjectToDatabaseTag(oSource, oDatabase, sVarName, nType, sTag); + + sqlquery q = _PrepareVariableInsert(oDatabase, nType, sVarName, sTag); + SqlBindJson(q, "@value", jValue); + SqlStep(q); + + if (bDelete) + _DeleteLocalVariable(oSource, sVarName, nType); + } + _CommitSQLTransaction(oDatabase); +} + +/// @private Copies specified variables from oSource (db) to oTarget (game object). +void _CopyVariablesToObject(object oDatabase, object oTarget, int nTypes, string sVarNames, + string sTag, int nTime, int bDelete) +{ + if (!GetIsPC(oDatabase) && oDatabase != DB_MODULE && oDatabase != DB_CAMPAIGN) + { + if (IsDebugging(DEBUG_LEVEL_NOTICE)) + Notice("Attempt to copy database variables to game object failed:" + + "\n oDatabase -> " + GetName(oDatabase) + + "\n oTarget -> " + GetName(oTarget) + + "\n nTypes -> " + IntToHexString(nTypes) + + "\n sVarName -> " + sVarNames + + "\n sTag -> " + sTag + + "\n nTime -> " + IntToString(nTime) + + "\n bDelete -> " + (bDelete ? "TRUE" : "FALSE")); + return; + } + + json jVariables = _DatabaseVariablesToJson(oDatabase, nTypes, sVarNames, sTag, nTime); + int nCount = JsonGetLength(jVariables); + + if (!nCount) + return; + + int n; for (n; n < nCount; n++) + { + json jVariable = JsonPointer(jVariables, "/" + IntToString(n)); + int nType = JsonGetInt(JsonPointer(jVariable, "/type")); + string sVarName = JsonGetString(JsonPointer(jVariable, "/varname")); + string sTag = JsonGetString(JsonPointer(jVariables, "/tag")); + json jValue = JsonPointer(jVariable, "/value"); + + _SetLocalVariable(oTarget, DatabaseToObjectVarName(oDatabase, oTarget, sVarName, sTag, nType), nType, jValue); + } + + if (bDelete) + SqlStep(_PrepareComplexVariableDelete(oDatabase, nTypes, sVarNames, sTag, nTime)); +} + +void _CopyDatabaseVariablesToDatabase(object oSource, object oTarget, int nTypes, string sVarNames, + string sTag, int nTime, int bDelete) +{ + if ((!GetIsPC(oSource) && oSource != DB_MODULE && oSource != DB_CAMPAIGN) || + (!GetIsPC(oTarget) && oTarget != DB_MODULE && oTarget != DB_CAMPAIGN) || + (oSource == oTarget)) + { + if (IsDebugging(DEBUG_LEVEL_NOTICE)) + Notice("Attempt to copy variables between databases failed:" + + "\n oSource -> " + GetName(oSource) + + "\n oTarget -> " + GetName(oTarget) + + "\n nTypes -> " + IntToHexString(nTypes) + + "\n sVarName -> " + sVarNames + + "\n sTag -> " + sTag + + "\n nTime -> " + IntToString(nTime) + + "\n bDelete -> " + (bDelete ? "TRUE" : "FALSE")); + return; + } + + json jVariables = _DatabaseVariablesToJson(oSource, nTypes, sVarNames, sTag, nTime); + + int bPC = GetIsPC(oTarget); + int bCampaign = oTarget == DB_CAMPAIGN; + string sTable = bPC ? VARIABLE_TABLE_PC : VARIABLE_TABLE_MODULE; + + string s = "INSERT INTO " + sTable + " (type, varname, value, tag, timestamp) " + + "SELECT value ->> '$.type', value ->> '$.varname', value ->> '$.value', " + + "value ->> '$.tag', strftime('%s','now') FROM (SELECT value FROM json_each(@variables));"; + + sqlquery q = bPC || !bCampaign ? _PrepareQueryObject(oTarget, s) : _PrepareQueryCampaign(s); + SqlBindJson(q, "@variables", jVariables); + SqlStep(q); + + if (bDelete) + SqlStep(_PrepareComplexVariableDelete(oSource, nTypes, sVarNames, sTag, nTime)); +} + +void _CopyLocalVariablesToObject(object oSource, object oTarget, int nTypes, + string sVarNames, int bDelete) +{ + if (oSource == GetModule()) + { + Notice("Attempt to copy variables between objects failed; " + + "cannot copy from the module object"); + return; + } + else if (oSource == oTarget) + return; + + json jVariables = _LocalVariablesToJson(oSource, nTypes, sVarNames); + int nCount = JsonGetLength(jVariables); + + if (!nCount) + return; + + int n; for (n; n < nCount; n++) + { + json jVariable = JsonPointer(jVariables, "/" + IntToString(n)); + int nType = JsonGetInt(JsonPointer(jVariable, "/type")); + string sVarName = JsonGetString(JsonPointer(jVariable, "/varname")); + json jValue = JsonPointer(jVariable, "/value"); + + _SetLocalVariable(oTarget, sVarName, nType, jValue); + + if (bDelete) + _DeleteLocalVariable(oSource, sVarName, nType); + } +} + +// ----------------------------------------------------------------------------- +// Public Functions +// ----------------------------------------------------------------------------- + +void CreateVariableTable(object oObject) +{ + string sVarName = VARIABLE_OBJECT; + string sTable = VARIABLE_TABLE_MODULE; + int bCampaign = oObject == DB_CAMPAIGN; + + if (bCampaign) + { + sVarName = VARIABLE_CAMPAIGN; + oObject = DB_MODULE; + } + else if (GetIsPC(oObject)) + sTable = VARIABLE_TABLE_PC; + else if (oObject != DB_MODULE) + return; + + if (GetLocalInt(oObject, sVarName)) + return; + + string s = "CREATE TABLE IF NOT EXISTS " + sTable + " (" + + "type INTEGER, " + + "varname TEXT, " + + "tag TEXT, " + + "value TEXT, " + + "timestamp INTEGER, " + + "PRIMARY KEY (type, varname, tag));"; + + sqlquery q; + if (bCampaign) + q = SqlPrepareQueryCampaign(VARIABLE_CAMPAIGN_DATABASE, s); + else + q = SqlPrepareQueryObject(oObject, s); + + SqlStep(q); + SetLocalInt(oObject, sVarName, TRUE); +} + +// ----------------------------------------------------------------------------- +// Local Variables +// ----------------------------------------------------------------------------- + +json GetLocalVariables(object oObject, int nType = VARIABLE_TYPE_ALL, string sVarName = "*") +{ + if (oObject == DB_MODULE) + return JsonArray(); + + json jVariables = _LocalVariablesToJson(oObject, nType, sVarName); + int nCount = JsonGetLength(jVariables); + + if (!nCount) + return JsonArray(); + + int n; for (n; n < nCount; n++) + { + json j = JsonArrayGet(jVariables, n); + j = JsonObjectSet(j, "type", JsonInt(_TypeToVariableType(JsonObjectGet(j, "type")))); + + jVariables = JsonArraySet(jVariables, n, j); + } + + return jVariables; +} + +void DeleteLocalVariables(object oObject, int nTypes = VARIABLE_TYPE_NONE, string sVarNames = "") +{ + json jVariables = GetLocalVariables(oObject, nTypes, sVarNames); + int n; for (n; n < JsonGetLength(jVariables); n++) + { + json jVariable = JsonArrayGet(jVariables, n); + int nType = JsonGetInt(JsonObjectGet(jVariable, "type")); + string sName = JsonGetString(JsonObjectGet(jVariable, "varname")); + + _DeleteLocalVariable(oObject, sName, nType); + } +} + +void CopyLocalVariablesToObject(object oSource, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", int bDelete = TRUE) +{ + _CopyLocalVariablesToObject(oSource, oTarget, nType, sVarName, bDelete); +} + +void CopyLocalVariablesToDatabase(object oSource, object oDatabase, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int bDelete = TRUE) +{ + _CopyVariablesToDatabase(oSource, oDatabase, nType, sVarName, sTag, bDelete); +} + +void CopyDatabaseVariablesToObject(object oDatabase, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0, int bDelete = TRUE) +{ + _CopyVariablesToObject(oDatabase, oTarget, nType, sVarName, sTag, nTime, bDelete); +} + +void CopyDatabaseVariablesToDatabase(object oSource, object oTarget, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0, int bDelete = TRUE) +{ + _CopyDatabaseVariablesToDatabase(oSource, oTarget, nType, sVarName, sTag, nTime, bDelete); +} + +int HasLocalInt(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_INT, sVarName)); +} + +int HasLocalFloat(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_FLOAT, sVarName)); +} + +int HasLocalString(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_STRING, sVarName)); +} + +int HasLocalObject(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_OBJECT, sVarName)); +} + +int HasLocalLocation(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_LOCATION, sVarName)); +} + +int HasLocalJson(object oObject, string sVarName) +{ + return JsonGetLength(_LocalVariablesToJson(oObject, VARIABLE_TYPE_JSON, sVarName)); +} + +// SetModule* ------------------------------------------------------------------ + +void SetModuleInt(string sVarName, int nValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nValue); + SqlStep(q); +} + +void SetModuleFloat(string sVarName, float fValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fValue); + SqlStep(q); +} + +void SetModuleString(string sVarName, string sValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_STRING, sVarName, sTag); + SqlBindString(q, "@value", sValue); + SqlStep(q); +} + +void SetModuleObject(string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_OBJECT, sVarName, sTag); + SqlBindString(q, "@value", ObjectToString(oValue)); + SqlStep(q); +} + +void SetModuleSerialized(string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + SqlBindJson(q, "@value", ObjectToJson(oValue, TRUE)); + SqlStep(q); +} + +void SetModuleLocation(string sVarName, location lValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_LOCATION, sVarName, sTag); + SqlBindJson(q, "@value", LocationToJson(lValue)); + SqlStep(q); +} + +void SetModuleVector(string sVarName, vector vValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_VECTOR, sVarName, sTag); + SqlBindJson(q, "@value", VectorToJson(vValue)); + SqlStep(q); +} + +void SetModuleJson(string sVarName, json jValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_MODULE, VARIABLE_TYPE_JSON, sVarName, sTag); + SqlBindJson(q, "@value", jValue); + SqlStep(q); +} + +void SetModuleVariableTag(int nType, string sVarName, string sTag = "", string sNewTag = "") +{ + SqlStep(_PrepareTagUpdate(DB_MODULE, nType, sVarName, sTag, sNewTag)); +} + +// GetModule* ------------------------------------------------------------------ + +int GetModuleInt(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float GetModuleFloat(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string GetModuleString(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object GetModuleObject(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +object GetModuleSerialized(string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID) +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + return SqlStep(q) ? JsonToObject(SqlGetJson(q, 0), l, oTarget, TRUE) : OBJECT_INVALID; +} + +location GetModuleLocation(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector GetModuleVector(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json GetModuleJson(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_MODULE, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +json GetModuleVariablesByPattern(int nType = VARIABLE_TYPE_ALL, string sVarName = "", + string sTag = "", int nTime = 0) +{ + return _DatabaseVariablesToJson(DB_MODULE, nType, sVarName, sTag, nTime); +} + +// DeleteModule* --------------------------------------------------------------- + +int DeleteModuleInt(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float DeleteModuleFloat(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string DeleteModuleString(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object DeleteModuleObject(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +void DeleteModuleSerialized(string sVarName, string sTag = "") +{ + SqlStep(_PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_SERIALIZED, sVarName, sTag)); +} + +location DeleteModuleLocation(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector DeleteModuleVector(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json DeleteModuleJson(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_MODULE, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +void DeleteModuleVariables() +{ + SqlStep(_PrepareComplexVariableDelete(DB_MODULE, VARIABLE_TYPE_NONE, "*", "*", 0)); +} + +void DeleteModuleVariablesByPattern(int nType = VARIABLE_TYPE_NONE, string sVarName = "", + string sTag = "*", int nTime = 0) +{ + SqlStep(_PrepareComplexVariableDelete(DB_MODULE, nType, sVarName, sTag, nTime)); +} + +int IncrementModuleInt(string sVarName, int nIncrement = 1, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(DB_MODULE, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nIncrement); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +int DecrementModuleInt(string sVarName, int nDecrement = -1, string sTag = "") +{ + if (nDecrement == 0) return GetModuleInt(sVarName); + else if (nDecrement > 0) nDecrement *= -1; + return IncrementModuleInt(sVarName, nDecrement, sTag); +} + +float IncrementModuleFloat(string sVarName, float fIncrement = 1.0, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(DB_MODULE, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fIncrement); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +float DecrementModuleFloat(string sVarName, float fDecrement = -1.0, string sTag = "") +{ + if (fDecrement == 0.0) return GetModuleFloat(sVarName); + else if (fDecrement > 0.0) fDecrement *= -1.0; + return IncrementModuleFloat(sVarName, fDecrement, sTag); +} + +string AppendModuleString(string sVarName, string sAppend, string sTag = "") +{ + sqlquery q = _PrepareVariableAppend(DB_MODULE, sVarName, sTag); + SqlBindString(q, "@value", sAppend); + SqlBindInt (q, "@type", VARIABLE_TYPE_STRING); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +// Player Database ------------------------------------------------------------- + +void SetPlayerInt(object oPlayer, string sVarName, int nValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nValue); + SqlStep(q); +} + +void SetPlayerFloat(object oPlayer, string sVarName, float fValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fValue); + SqlStep(q); +} + +void SetPlayerString(object oPlayer, string sVarName, string sValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_STRING, sVarName, sTag); + SqlBindString(q, "@value", sValue); + SqlStep(q); +} + +void SetPlayerObject(object oPlayer, string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_OBJECT, sVarName, sTag); + SqlBindString(q, "@value", ObjectToString(oValue)); + SqlStep(q); +} + +void SetPlayerSerialized(object oPlayer, string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + SqlBindJson(q, "@value", ObjectToJson(oValue, TRUE)); + SqlStep(q); +} + +void SetPlayerLocation(object oPlayer, string sVarName, location lValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_LOCATION, sVarName, sTag); + SqlBindJson(q, "@value", LocationToJson(lValue)); + SqlStep(q); +} + +void SetPlayerVector(object oPlayer, string sVarName, vector vValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_VECTOR, sVarName, sTag); + SqlBindJson(q, "@value", VectorToJson(vValue)); + SqlStep(q); +} + +void SetPlayerJson(object oPlayer, string sVarName, json jValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(oPlayer, VARIABLE_TYPE_JSON, sVarName, sTag); + SqlBindJson(q, "@value", jValue); + SqlStep(q); +} + +// GetPlayer* ------------------------------------------------------------------ + +int GetPlayerInt(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float GetPlayerFloat(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string GetPlayerString(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object GetPlayerObject(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +object GetPlayerSerialized(object oPlayer, string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID) +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + return SqlStep(q) ? JsonToObject(SqlGetJson(q, 0), l, oTarget, TRUE) : OBJECT_INVALID; +} + +location GetPlayerLocation(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector GetPlayerVector(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json GetPlayerJson(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(oPlayer, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +json GetPlayerVariablesByPattern(object oPlayer, int nType = VARIABLE_TYPE_ALL, + string sVarName = "", string sTag = "", int nTime = 0) +{ + return _DatabaseVariablesToJson(oPlayer, nType, sVarName, sTag, nTime); +} + +// DeletePlayer* --------------------------------------------------------------- + +int DeletePlayerInt(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float DeletePlayerFloat(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string DeletePlayerString(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object DeletePlayerObject(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +void DeletePlayerSerialized(object oPlayer, string sVarName, string sTag = "") +{ + SqlStep(_PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_SERIALIZED, sVarName, sTag)); +} + +location DeletePlayerLocation(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector DeletePlayerVector(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json DeletePlayerJson(object oPlayer, string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(oPlayer, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +void DeletePlayerVariables(object oPlayer) +{ + SqlStep(_PrepareComplexVariableDelete(oPlayer, 0, "*", "*", 0)); +} + +void DeletePlayerVariablesByPattern(object oPlayer, int nType = VARIABLE_TYPE_NONE, + string sVarName = "", string sTag = "", int nTime = 0) +{ + SqlStep(_PrepareComplexVariableDelete(oPlayer, nType, sVarName, sTag, nTime)); +} + +int IncrementPlayerInt(object oPlayer, string sVarName, int nIncrement = 1, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(oPlayer, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nIncrement); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +int DecrementPlayerInt(object oPlayer, string sVarName, int nDecrement = -1, string sTag = "") +{ + if (nDecrement == 0) return GetPlayerInt(oPlayer, sVarName); + else if (nDecrement > 0) nDecrement *= -1; + return IncrementPlayerInt(oPlayer, sVarName, nDecrement, sTag); +} + +float IncrementPlayerFloat(object oPlayer, string sVarName, float fIncrement = 1.0, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(oPlayer, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fIncrement); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +float DecrementPlayerFloat(object oPlayer, string sVarName, float fDecrement = -1.0, string sTag = "") +{ + if (fDecrement == 0.0) return GetPlayerFloat(oPlayer, sVarName); + else if (fDecrement > 0.0) fDecrement *= -1.0; + return IncrementPlayerFloat(oPlayer, sVarName, fDecrement, sTag); +} + +string AppendPlayerString(object oPlayer, string sVarName, string sAppend, string sTag = "") +{ + sqlquery q = _PrepareVariableAppend(oPlayer, sVarName, sTag); + SqlBindString(q, "@value", sAppend); + SqlBindInt (q, "@type", VARIABLE_TYPE_STRING); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +// SetPersistent* ------------------------------------------------------------------ + +void SetPersistentInt(string sVarName, int nValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nValue); + SqlStep(q); +} + +void SetPersistentFloat(string sVarName, float fValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fValue); + SqlStep(q); +} + +void SetPersistentString(string sVarName, string sValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_STRING, sVarName, sTag); + SqlBindString(q, "@value", sValue); + SqlStep(q); +} + +void SetPersistentObject(string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_OBJECT, sVarName, sTag); + SqlBindString(q, "@value", ObjectToString(oValue)); + SqlStep(q); +} + +void SetPersistentSerialized(string sVarName, object oValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + SqlBindJson(q, "@value", ObjectToJson(oValue, TRUE)); + SqlStep(q); +} + +void SetPersistentLocation(string sVarName, location lValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_LOCATION, sVarName, sTag); + SqlBindJson(q, "@value", LocationToJson(lValue)); + SqlStep(q); +} + +void SetPersistentVector(string sVarName, vector vValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_VECTOR, sVarName, sTag); + SqlBindJson(q, "@value", VectorToJson(vValue)); + SqlStep(q); +} + +void SetPersistentJson(string sVarName, json jValue, string sTag = "") +{ + sqlquery q = _PrepareVariableInsert(DB_CAMPAIGN, VARIABLE_TYPE_JSON, sVarName, sTag); + SqlBindJson(q, "@value", jValue); + SqlStep(q); +} + +void SetPersistentVariableTag(int nType, string sVarName, string sTag = "", string sNewTag = "") +{ + SqlStep(_PrepareTagUpdate(DB_CAMPAIGN, nType, sVarName, sTag, sNewTag)); +} + +// GetPersistent* ------------------------------------------------------------------ + +int GetPersistentInt(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float GetPersistentFloat(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string GetPersistentString(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object GetPersistentObject(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +object GetPersistentSerialized(string sVarName, string sTag, location l, object oTarget = OBJECT_INVALID) +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_SERIALIZED, sVarName, sTag); + return SqlStep(q) ? JsonToObject(SqlGetJson(q, 0), l, oTarget, TRUE) : OBJECT_INVALID; +} + +location GetPersistentLocation(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector GetPersistentVector(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json GetPersistentJson(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareVariableSelect(DB_CAMPAIGN, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +json GetPersistentVariablesByPattern(int nType = VARIABLE_TYPE_ALL, string sVarName = "*", + string sTag = "*", int nTime = 0) +{ + return _DatabaseVariablesToJson(DB_CAMPAIGN, nType, sVarName, sTag, nTime); +} + +// DeletePersistent* --------------------------------------------------------------- + +int DeletePersistentInt(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_INT, sVarName, sTag); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +float DeletePersistentFloat(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_FLOAT, sVarName, sTag); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +string DeletePersistentString(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_STRING, sVarName, sTag); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} + +object DeletePersistentObject(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_OBJECT, sVarName, sTag); + return SqlStep(q) ? StringToObject(SqlGetString(q, 0)) : OBJECT_INVALID; +} + +void DeletePersistentSerialized(string sVarName, string sTag = "") +{ + SqlStep(_PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_SERIALIZED, sVarName, sTag)); +} + +location DeletePersistentLocation(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_LOCATION, sVarName, sTag); + return SqlStep(q) ? JsonToLocation(SqlGetJson(q, 0)) : Location(OBJECT_INVALID, Vector(), 0.0); +} + +vector DeletePersistentVector(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_VECTOR, sVarName, sTag); + + vector v; + if (SqlStep(q)) v = JsonToVector(SqlGetJson(q, 0)); + else v = Vector(); + + return v; +} + +json DeletePersistentJson(string sVarName, string sTag = "") +{ + sqlquery q = _PrepareSimpleVariableDelete(DB_CAMPAIGN, VARIABLE_TYPE_JSON, sVarName, sTag); + return SqlStep(q) ? SqlGetJson(q, 0) : JsonNull(); +} + +void DeletePersistentVariables() +{ + SqlStep(_PrepareComplexVariableDelete(DB_CAMPAIGN, 0, "*", "*", 0)); +} + +void DeletePersistentVariablesByPattern(int nType = VARIABLE_TYPE_NONE, string sVarName = "", + string sTag = "", int nTime = 0) +{ + SqlStep(_PrepareComplexVariableDelete(DB_CAMPAIGN, nType, sVarName, sTag, nTime)); +} + +int IncrementPersistentInt(string sVarName, int nIncrement = 1, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(DB_CAMPAIGN, VARIABLE_TYPE_INT, sVarName, sTag); + SqlBindInt(q, "@value", nIncrement); + return SqlStep(q) ? SqlGetInt(q, 0) : 0; +} + +int DecrementPersistentInt(string sVarName, int nDecrement = -1, string sTag = "") +{ + if (nDecrement == 0) return GetPersistentInt(sVarName); + else if (nDecrement > 0) nDecrement *= -1; + return IncrementPersistentInt(sVarName, nDecrement, sTag); +} + +float IncrementPersistentFloat(string sVarName, float fIncrement = 1.0, string sTag = "") +{ + sqlquery q = _PrepareVariableIncrement(DB_CAMPAIGN, VARIABLE_TYPE_FLOAT, sVarName, sTag); + SqlBindFloat(q, "@value", fIncrement); + return SqlStep(q) ? SqlGetFloat(q, 0) : 0.0; +} + +float DecrementPersistentFloat(string sVarName, float fDecrement = -1.0, string sTag = "") +{ + if (fDecrement == 0.0) return GetPersistentFloat(sVarName); + else if (fDecrement > 0.0) fDecrement *= -1.0; + return IncrementPersistentFloat(sVarName, fDecrement, sTag); +} + +string AppendPersistentString(string sVarName, string sAppend, string sTag = "") +{ + sqlquery q = _PrepareVariableAppend(DB_CAMPAIGN, sVarName, sTag); + SqlBindString(q, "@value", sAppend); + SqlBindInt (q, "@type", VARIABLE_TYPE_STRING); + return SqlStep(q) ? SqlGetString(q, 0) : ""; +} diff --git a/_module/nss/util_i_varlists.nss b/_module/nss/util_i_varlists.nss new file mode 100644 index 0000000..6844d54 --- /dev/null +++ b/_module/nss/util_i_varlists.nss @@ -0,0 +1,2158 @@ +/// ---------------------------------------------------------------------------- +/// @file util_i_varlists.nss +/// @author Ed Burke (tinygiant98) +/// @brief Functions for manipulating local variable lists. +/// @details +/// Local variable lists are json arrays of a single type stored as local +/// variables. They are namespaced by type, so you can maintain lists of +/// different types using the same varname. +/// +/// The majority of functions in this file apply to each possible variable type: +/// float, int, location, vector, object, string, json. However, there are some +/// that only apply to a subset of variable types, such as +/// Sort[Float|Int|String]List() and [Increment|Decrement]ListInt(). +/// ---------------------------------------------------------------------------- + +#include "util_i_math" + +// ----------------------------------------------------------------------------- +// Constants +// ----------------------------------------------------------------------------- + +// Constants used to describe float|int|string sorting order +const int LIST_SORT_ASC = 1; +const int LIST_SORT_DESC = 2; + +// Prefixes used to keep list variables from colliding with other locals. These +// constants are considered private and should not be referenced from other scripts. +const string LIST_REF = "Ref:"; +const string VARLIST_TYPE_VECTOR = "VL:"; +const string VARLIST_TYPE_FLOAT = "FL:"; +const string VARLIST_TYPE_INT = "IL:"; +const string VARLIST_TYPE_LOCATION = "LL:"; +const string VARLIST_TYPE_OBJECT = "OL:"; +const string VARLIST_TYPE_STRING = "SL:"; +const string VARLIST_TYPE_JSON = "JL:"; + +// ----------------------------------------------------------------------------- +// Function Prototypes +// ----------------------------------------------------------------------------- + +/// @brief Convert a vector to a json object. +/// @param vPosition The vector to convert. +/// @note Alias for JsonVector(). +json VectorToJson(vector vPosition = [0.0, 0.0, 0.0]); + +/// @brief Convert a vector to a json object. +/// @param vPosition The vector to convert. +json JsonVector(vector vPosition = [0.0, 0.0, 0.0]); + +/// @brief Convert a json object to a vector. +/// @param jPosition The json object to convert. +/// @note Alias for JsonGetVector(). +vector JsonToVector(json jPosition); + +/// @brief Convert a json object to a vector. +/// @param jPosition The json object to convert. +vector JsonGetVector(json jPosition); + +/// @brief Convert a location to a json object. +/// @param lLocation The location to convert. +/// @note Alias for JsonLocation(). +json LocationToJson(location lLocation); + +/// @brief Convert a location to a json object. +/// @param lLocation The location to convert. +json JsonLocation(location lLocation); + +/// @brief Convert a json object to a location. +/// @param jLocation The json object to convert. +/// @note Alias for JsonGetLocation(). +location JsonToLocation(json jLocation); + +/// @brief Convert a json object to a location. +/// @param jLocation The json object to convert. +location JsonGetLocation(json jLocation); + +/// @brief Add a value to a float list on a target. +/// @param oTarget The object the list is stored on. +/// @param fValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListFloat(object oTarget, float fValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to an int list on a target. +/// @param oTarget The object the list is stored on. +/// @param nValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListInt(object oTarget, int nValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to a location list on a target. +/// @param oTarget The object the list is stored on. +/// @param lValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListLocation(object oTarget, location lValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to a vector list on a target. +/// @param oTarget The object the list is stored on. +/// @param vValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListVector(object oTarget, vector vValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to an object list on a target. +/// @param oTarget The object the list is stored on. +/// @param oValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListObject(object oTarget, object oValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to a string list on a target. +/// @param oTarget The object the list is stored on. +/// @param sValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListString(object oTarget, string sValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Add a value to a json list on a target. +/// @param oTarget The object the list is stored on. +/// @param jValue The value to add to the list. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, will not add the value if it is already present. +/// @returns TRUE if the operation was successful; FALSE otherwise. +int AddListJson(object oTarget, json jValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Return the value at an index in a target's float list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns 0.0 if no value is found at nIndex. +float GetListFloat(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns 0 if no value is found at nIndex. +int GetListInt(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's location list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns LOCATION_INVALID if no value is found at nIndex. +location GetListLocation(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's vector list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns [0.0. 0.0, 0.0] if no value was found at nIndex. +vector GetListVector(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's object list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns OBJECT_INVALID if no value was found at nIndex. +object GetListObject(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's string list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns "" if no value was found at nIndex. +string GetListString(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Return the value at an index in a target's json list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @returns JSON_NULL if no value was found at nIndex. +json GetListJson(object oTarget, int nIndex = 0, string sListName = ""); + +/// @brief Delete the value at an index on an object's float list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListFloat(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListInt(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's location list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListLocation(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListVector(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's object list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListObject(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's string list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListString(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Delete the value at an index on an object's json list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int DeleteListJson(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's float list. +/// @param oTarget The object the list is stored on. +/// @param fValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListFloat(object oTarget, float fValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListInt(object oTarget, int nValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's location list. +/// @param oTarget The object the list is stored on. +/// @param lValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListLocation(object oTarget, location lValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param vValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListVector(object oTarget, vector vValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's object list. +/// @param oTarget The object the list is stored on. +/// @param oValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListObject(object oTarget, object oValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListString(object oTarget, string sValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Remove the first instance of a value from an object's json list. +/// @param oTarget The object the list is stored on. +/// @param jValue The value to remove. +/// @param sListName The name of the list. +/// @param bMaintainOrder Not used; exists for legacy purposes only. +/// @returns The number of items remanining in the list. +int RemoveListJson(object oTarget, json jValue, string sListName = "", int bMaintainOrder = FALSE); + +/// @brief Removes and returns the first value from an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +float PopListFloat(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's int list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int PopListInt(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's location list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +location PopListLocation(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +vector PopListVector(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's object list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +object PopListObject(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +string PopListString(object oTarget, string sListName = ""); + +/// @brief Removes and returns the first value from an object's json list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +json PopListJson(object oTarget, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// float list. +/// @param oTarget The object the list is stored on. +/// @param fValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListFloat(object oTarget, float fValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// int list. +/// @param oTarget The object the list is stored on. +/// @param nValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListInt(object oTarget, int nValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// location list. +/// @param oTarget The object the list is stored on. +/// @param lValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListLocation(object oTarget, location lValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// vector list. +/// @param oTarget The object the list is stored on. +/// @param vValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListVector(object oTarget, vector vValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// object list. +/// @param oTarget The object the list is stored on. +/// @param oValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListObject(object oTarget, object oValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// string list. +/// @param oTarget The object the list is stored on. +/// @param sValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListString(object oTarget, string sValue, string sListName = ""); + +/// @brief Return the index of the first occurrence of a value in an object's +/// json list. +/// @param oTarget The object the list is stored on. +/// @param jValue The value to find. +/// @param sListName The name of the list. +/// @returns The index of the value (0-based), or -1 if it is not in the list. +int FindListJson(object oTarget, json jValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param fValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListFloat(object oTarget, float fValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListInt(object oTarget, int nValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param lValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListLocation(object oTarget, location lValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param vValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListVector(object oTarget, vector vValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's object list. +/// @param oTarget The object the list is stored on. +/// @param oValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListObject(object oTarget, object oValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListString(object oTarget, string sValue, string sListName = ""); + +/// @brief Return whether a value is present in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param jValue The value to find. +/// @param sListName The name of the list. +/// @returns TRUE if the value is in the list; FALSE otherwise. +int HasListJson(object oTarget, json jValue, string sListName = ""); + +/// @brief Insert a value at an index in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param fValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListFloat(object oTarget, int nIndex, float fValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param nValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListInt(object oTarget, int nIndex, int nValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param lValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListLocation(object oTarget, int nIndex, location lValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param vValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListVector(object oTarget, int nIndex, vector vValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's objeect list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param oValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListObject(object oTarget, int nIndex, object oValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param sValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListString(object oTarget, int nIndex, string sValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Insert a value at an index in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to insert the value at. If the index exceeds the +/// length of the list, nothing is added. +/// @param jValue The value to insert. +/// @param sListName The name of the list. +/// @param bAddUnique If TRUE, the insert operation will be conducted first and +/// then duplicate values will be removed. +/// @returns The length of the updated list. +int InsertListJson(object oTarget, int nIndex, json jValue, string sListName = "", int bAddUnique = FALSE); + +/// @brief Set the value at an index in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param fValue The value to set. +/// @param sListName The name of the list. +void SetListFloat(object oTarget, int nIndex, float fValue, string sListName = ""); + +/// @brief Set the value at an index in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param nValue The value to set. +/// @param sListName The name of the list. +void SetListInt(object oTarget, int nIndex, int nValue, string sListName = ""); + +/// @brief Set the value at an index in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param lValue The value to set. +/// @param sListName The name of the list. +void SetListLocation(object oTarget, int nIndex, location lValue, string sListName = ""); + +/// @brief Set the value at an index in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param vValue The value to set. +/// @param sListName The name of the list. +void SetListVector(object oTarget, int nIndex, vector vValue, string sListName = ""); + +/// @brief Set the value at an index in an object's object list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param oValue The value to set. +/// @param sListName The name of the list. +void SetListObject(object oTarget, int nIndex, object oValue, string sListName = ""); + +/// @brief Set the value at an index in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param sValue The value to set. +/// @param sListName The name of the list. +void SetListString(object oTarget, int nIndex, string sValue, string sListName = ""); + +/// @brief Set the value at an index in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index to set the value of. If the index exceeds the length +/// of the list, nothing is added. +/// @param jValue The value to set. +/// @param sListName The name of the list. +void SetListJson(object oTarget, int nIndex, json jValue, string sListName = ""); + +/// @brief Copy value from one object's float list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListFloat(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's int list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListInt(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's location list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListLocation(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's vector list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListVector(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's object list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListObject(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's string list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListString(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Copy value from one object's json list to another's. +/// @param oSource The object to copy the list values of. +/// @param oTarget The object to copy the list values to. +/// @param sSourceName The name of the list on oSource. +/// @param sTargetName The name of the list on oTarget. +/// @param nIndex The index to begin copying from. +/// @param nRange The number of values to copy. If -1, will copy all values from +/// nIndex and up. +/// @param bAddUnique If TRUE, the copy operation will be conducted first and +/// then any duplicate values will be removed. Values in the target list +/// will be prioritiezed over values from the source list. +/// @returns The number of values copied. +int CopyListJson(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE); + +/// @brief Increment the value at an index in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param nIncrement The amount to increment the value by. +/// @param sListName The name of the list. +/// @returns The new value of the int. +int IncrementListInt(object oTarget, int nIndex, int nIncrement = 1, string sListName = ""); + +/// @brief Decrement the value at an index in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nIndex The index of the value. +/// @param nIncrement The amount to decrement the value by. +/// @param sListName The name of the list. +/// @returns The new value of the int. +int DecrementListInt(object oTarget, int nIndex, int nDecrement = -1, string sListName = ""); + +/// @brief Convert an object's float list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with JsonGetFloat(). +json GetFloatList(object oTarget, string sListName = ""); + +/// @brief Convert an object's int list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with JsonGetInt(). +json GetIntList(object oTarget, string sListName = ""); + +/// @brief Convert an object's location list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with JsonGetLocation(). +json GetLocationList(object oTarget, string sListName = ""); + +/// @brief Convert an object's vector list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with JsonGetVector(). +json GetVectorList(object oTarget, string sListName = ""); + +/// @brief Convert an object's object list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with +/// ObjectToString(JsonGetString()). +json GetObjectList(object oTarget, string sListName = ""); + +/// @brief Convert an object's string list to a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +/// @note Elements of the returned array can be decoded with JsonGetString(). +json GetStringList(object oTarget, string sListName = ""); + +/// @brief Convert an object's json list into a json array. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +json GetJsonList(object oTarget, string sListName = ""); + +/// @brief Save a json array as an object's float list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonFloat()s. +/// @param sListName The name of the list. +void SetFloatList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's int list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonInt()s. +/// @param sListName The name of the list. +void SetIntList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's location list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonLocation()s. +/// @param sListName The name of the list. +void SetLocationList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonVector()s. +/// @param sListName The name of the list. +void SetVectorList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's object list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonObject()s. +/// @param sListName The name of the list. +void SetObjectList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's string list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of JsonString()s. +/// @param sListName The name of the list. +void SetStringList(object oTarget, json jList, string sListName = ""); + +/// @brief Save a json array as an object's json list. +/// @param oTarget The object the list is stored on. +/// @param jList A JsonArray() made up of any json types. +/// @param sListName The name of the list. +void SetJsonList(object oTarget, json jList, string sListName = ""); + +/// @brief Delete an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteFloatList(object oTarget, string sListName = ""); + +/// @brief Delete an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteIntList(object oTarget, string sListName = ""); + +/// @brief Delete an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteLocationList(object oTarget, string sListName = ""); + +/// @brief Delete an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteVectorList(object oTarget, string sListName = ""); + +/// @brief Delete an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteObjectList(object oTarget, string sListName = ""); + +/// @brief Delete an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteStringList(object oTarget, string sListName = ""); + +/// @brief Delete an object's json list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void DeleteJsonList(object oTarget, string sListName = ""); + +/// @brief Create a float list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @param fDefault The value to initialize the list with. +/// @returns A json array copy of the created list. +json DeclareFloatList(object oTarget, int nCount, string sListName = "", float fDefault = 0.0); + +/// @brief Create an int list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @param nDefault The value to initialize the list with. +/// @returns A json array copy of the created list. +json DeclareIntList(object oTarget, int nCount, string sListName = "", int nDefault = 0); + +/// @brief Create a location list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @returns A json array copy of the created list. +json DeclareLocationList(object oTarget, int nCount, string sListName = ""); + +/// @brief Create a vector list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @returns A json array copy of the created list. +json DeclareVectorList(object oTarget, int nCount, string sListName = ""); + +/// @brief Create an object list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @returns A json array copy of the created list. +json DeclareObjectList(object oTarget, int nCount, string sListName = ""); + +/// @brief Create a string list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @param sDefault The value to initialize the list with. +/// @returns A json array copy of the created list. +json DeclareStringList(object oTarget, int nCount, string sListName = "", string sDefault = ""); + +/// @brief Create a json list on a target, deleting any current list. +/// @param oTarget The object to create the list on. +/// @param nCount The number of values to initialize the list with. +/// @param sListName The name of the list. +/// @returns A json array copy of the created list. +json DeclareJsonList(object oTarget, int nCount, string sListName = ""); + +/// @brief Set the length of an object's float list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @param fDefault The value to set any added elements to. +/// @returns A json array copy of the updated list. +json NormalizeFloatList(object oTarget, int nCount, string sListName = "", float fDefault = 0.0); + +/// @brief Set the length of an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @param nDefault The value to set any added elements to. +/// @returns A json array copy of the updated list. +json NormalizeIntList(object oTarget, int nCount, string sListName = "", int nDefault = 0); + +/// @brief Set the length of an object's location list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @returns A json array copy of the updated list. +json NormalizeLocationList(object oTarget, int nCount, string sListName = ""); + +/// @brief Set the length of an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @returns A json array copy of the updated list. +json NormalizeVectorList(object oTarget, int nCount, string sListName = ""); + +/// @brief Set the length of an object's object list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @returns A json array copy of the updated list. +json NormalizeObjectList(object oTarget, int nCount, string sListName = ""); + +/// @brief Set the length of an object's string list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional values will be added to the end of the list. +/// @param sListName The name of the list. +/// @param sDefault The value to set any added elements to. +/// @returns A json array copy of the updated list. +json NormalizeStringList(object oTarget, int nCount, string sListName = "", string sDefault = ""); + +/// @brief Set the length of an object's json list. +/// @param oTarget The object the list is stored on. +/// @param nCount The length to set the list to. If less than the current +/// length, the list will be shortened to match. If greater than the current +/// length, additional null values will be added to the end of the list. +/// @param sListName The name of the list. +/// @returns A json array copy of the updated list. +json NormalizeJsonList(object oTarget, int nCount, string sListName = ""); + +/// @brief Copy all items from one object's float list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyFloatList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's int list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyIntList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's location list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyLocationList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's vector list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyVectorList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's object list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyObjectList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's string list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyStringList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Copy all items from one object's json list to another's. +/// @param oSource The object to copy the list from. +/// @param oTarget The object to copy the list to. +/// @param sSourceName The name of the source list. +/// @param sTargetName The name of the target list. +/// @param bAddUnique If TRUE, will only copy items that are not already present +/// in the target list. +void CopyJsonList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE); + +/// @brief Return the number of items in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountFloatList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountIntList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountLocationList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountVectorList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's object list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountObjectList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountStringList(object oTarget, string sListName = ""); + +/// @brief Return the number of items in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +int CountJsonList(object oTarget, string sListName = ""); + +/// @brief Sort an object's float list. +/// @param oTarget The object the list is stored on. +/// @param nOrder A `LIST_ORDER_*` constant representing how to sort the list. +/// @param sListName The name of the list. +void SortFloatList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = ""); + +/// @brief Sort an object's int list. +/// @param oTarget The object the list is stored on. +/// @param nOrder A `LIST_ORDER_*` constant representing how to sort the list. +/// @param sListName The name of the list. +void SortIntList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = ""); + +/// @brief Sort an object's string list. +/// @param oTarget The object the list is stored on. +/// @param nOrder A `LIST_ORDER_*` constant representing how to sort the list. +/// @param sListName The name of the list. +void SortStringList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = ""); + +/// @brief Shuffle the items in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleFloatList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleIntList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleLocationList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleVectorList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's object list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleObjectList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleStringList(object oTarget, string sListName = ""); + +/// @brief Shuffle the items in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ShuffleJsonList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's float list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseFloatList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's int list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseIntList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's location list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseLocationList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's vector list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseVectorList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's object list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseObjectList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's string list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseStringList(object oTarget, string sListName = ""); + +/// @brief Reverse the order of the items in an object's json list. +/// @param oTarget The object the list is stored on. +/// @param sListName The name of the list. +void ReverseJsonList(object oTarget, string sListName = ""); + +// ----------------------------------------------------------------------------- +// Function Implementations +// ----------------------------------------------------------------------------- + +// ----------------------------------------------------------------------------- +// Private Functions +// ----------------------------------------------------------------------------- + +// Determines whether nIndex is a valid reference to an array element in jList. +// If bNegative is TRUE, -1 will be returned as a valid nIndex value. +int _GetIsIndexValid(json jList, int nIndex, int bNegative = FALSE) +{ + return nIndex == 0 || nIndex >= (0 - bNegative) && nIndex < JsonGetLength(jList); +} + +// Retrieves json array sListName of sListType from oTarget. +json _GetList(object oTarget, string sListType, string sListName = "") +{ + json jList = GetLocalJson(oTarget, LIST_REF + sListType + sListName); + return jList == JSON_NULL ? JSON_ARRAY : jList; +} + +// Sets sListType json array jList as sListName on oTarget. +void _SetList(object oTarget, string sListType, string sListName, json jList) +{ + SetLocalJson(oTarget, LIST_REF + sListType + sListName, jList); +} + +// Deletes sListType json array sListName from oTarget. +void _DeleteList(object oTarget, string sListType, string sListName) +{ + DeleteLocalJson(oTarget, LIST_REF + sListType + sListName); +} + +// Inserts array element jValue into json array sListName at nIndex on oTarget. +// Returns the number of elements in the array after insertion. If bUnique is +// TRUE, duplicate values with be removed after the insert operation. +int _InsertListElement(object oTarget, string sListType, string sListName, + json jValue, int nIndex, int bUnique) +{ + json jList = _GetList(oTarget, sListType, sListName); + + if (_GetIsIndexValid(jList, nIndex, TRUE) == TRUE) + { + JsonArrayInsertInplace(jList, jValue, nIndex); + if (bUnique == TRUE) + jList = JsonArrayTransform(jList, JSON_ARRAY_UNIQUE); + + _SetList(oTarget, sListType, sListName, jList); + } + + return JsonGetLength(jList); +} + +// Returns array element at nIndex from array sListName on oTarget. If not +// found, returns JSON_NULL. +json _GetListElement(object oTarget, string sListType, string sListName, int nIndex) +{ + json jList = _GetList(oTarget, sListType, sListName); + return _GetIsIndexValid(jList, nIndex) ? JsonArrayGet(jList, nIndex) : JSON_NULL; +} + +// Deletes array element at nIndex from array sListName on oTarget. Element order +// is maintained. Returns the number of array elements remaining after deletion. +int _DeleteListElement(object oTarget, string sListType, string sListName, int nIndex) +{ + json jList = _GetList(oTarget, sListType, sListName); + + if (_GetIsIndexValid(jList, nIndex) == TRUE && JsonGetLength(jList) > 0) + { + JsonArrayDelInplace(jList, nIndex); + _SetList(oTarget, sListType, sListName, jList); + } + + return JsonGetLength(jList); +} + +// Finds array element jValue in array sListName on oTarget. If found, returns the +// index of the elements. If not, returns -1. +int _FindListElement(object oTarget, string sListType, string sListName, json jValue) +{ + json jList = _GetList(oTarget, sListType, sListName); + json jIndex = JsonFind(jList, jValue, 0, JSON_FIND_EQUAL); + return jIndex == JSON_NULL ? -1 : JsonGetInt(jIndex); +} + +// Deletes array element jValue from array sListName on oTarget. Element order +// is maintained. Returns the number of array elements remaining after deletion. +int _RemoveListElement(object oTarget, string sListType, string sListName, json jValue) +{ + json jList = _GetList(oTarget, sListType, sListName); + int nIndex = _FindListElement(oTarget, sListType, sListName, jValue); + + if (nIndex > -1) + { + JsonArrayDelInplace(jList, nIndex); + _SetList(oTarget, sListType, sListName, JsonArrayDel(jList, nIndex)); + } + + return JsonGetLength(jList); +} + +// Finds array element jValue in array sListName on oTarget. Returns TRUE if found, +// FALSE otherwise. +int _HasListElement(object oTarget, string sListType, string sListName, json jValue) +{ + return _FindListElement(oTarget, sListType, sListName, jValue) > -1; +} + +// Replaces array element at nIndex in array sListName on oTarget with jValue. +void _SetListElement(object oTarget, string sListType, string sListName, int nIndex, json jValue) +{ + json jList = _GetList(oTarget, sListType, sListName); + + if (_GetIsIndexValid(jList, nIndex) == TRUE) + _SetList(oTarget, sListType, sListName, JsonArraySet(jList, nIndex, jValue)); +} + +// This procedure exists because current json operations cannot easily append a list without +// removing duplicate elements or auto-sorting the list. BD is expected to update json +// functions with an append option. If so, replace this function with the json append +// function from nwscript.nss or fold this into _SortList() below. +json _JsonArrayAppend(json jFrom, json jTo) +{ + string sFrom = JsonDump(jFrom); + string sTo = JsonDump(jTo); + + sFrom = GetStringRight(sFrom, GetStringLength(sFrom) - 1); + sTo = GetStringLeft(sTo, GetStringLength(sTo) - 1); + + int nFrom = JsonGetLength(jFrom); + int nTo = JsonGetLength(jTo); + + string s = (nTo == 0 ? "" : + nTo > 0 && nFrom == 0 ? "" : ","); + + return JsonParse(sTo + s + sFrom); +} + +// Copies specified elements from oSource array sSourceName to oTarget array sTargetName. +// Copied elements start at nIndex and continue for nRange elements. Elements copied from +// oSource are appended to the end of oTarget's array. +int _CopyListElements(object oSource, object oTarget, string sListType, string sSourceName, + string sTargetName, int nIndex, int nRange, int bUnique) +{ + json jSource = _GetList(oSource, sListType, sSourceName); + json jTarget = _GetList(oTarget, sListType, sTargetName); + + if (jTarget == JSON_NULL) + jTarget = JSON_ARRAY; + + int nSource = JsonGetLength(jSource); + int nTarget = JsonGetLength(jTarget); + + if (nSource == 0) return 0; + + json jCopy, jReturn; + + if (nIndex == 0 && (nRange == -1 || nRange >= nSource)) + { + if (jSource == JSON_NULL || nSource == 0) + return 0; + + jReturn = _JsonArrayAppend(jSource, jTarget); + if (bUnique == TRUE) + jReturn = JsonArrayTransform(jReturn, JSON_ARRAY_UNIQUE); + + _SetList(oTarget, sListType, sTargetName, jReturn); + return nSource; + } + + if (_GetIsIndexValid(jSource, nIndex) == TRUE) + { + int nMaxIndex = nSource - nIndex; + if (nRange == -1) + nRange = nMaxIndex; + else if (nRange > (nMaxIndex)) + nRange = clamp(nRange, 1, nMaxIndex); + + jCopy = JsonArrayGetRange(jSource, nIndex, nIndex + (nRange - 1)); + jReturn = _JsonArrayAppend(jTarget, jCopy); + if (bUnique == TRUE) + jReturn = JsonArrayTransform(jReturn, JSON_ARRAY_UNIQUE); + + _SetList(oTarget, sListType, sTargetName, jReturn); + return JsonGetLength(jCopy) - JsonGetLength(JsonSetOp(jCopy, JSON_SET_INTERSECT, jTarget)); + } + + return 0; +} + +// Modifies an int list element by nIncrement and returns the new value. +int _IncrementListElement(object oTarget, string sListName, int nIndex, int nIncrement) +{ + json jList = _GetList(oTarget, VARLIST_TYPE_INT, sListName); + + if (_GetIsIndexValid(jList, nIndex)) + { + int nValue = JsonGetInt(JsonArrayGet(jList, nIndex)) + nIncrement; + JsonArraySetInplace(jList, nIndex, JsonInt(nValue)); + _SetList(oTarget, VARLIST_TYPE_INT, sListName, jList); + + return nValue; + } + + return 0; +} + +// Creates an array of length nLength jDefault elements as sListName on oTarget. +json _DeclareList(object oTarget, string sListType, string sListName, int nLength, json jDefault) +{ + json jList = JSON_ARRAY; + + int n; + for (n = 0; n < nLength; n++) + JsonArrayInsertInplace(jList, jDefault); + + _SetList(oTarget, sListType, sListName, jList); + return jList; +} + +// Sets the array length to nLength, adding/removing elements as required. +json _NormalizeList(object oTarget, string sListType, string sListName, int nLength, json jDefault) +{ + json jList = _GetList(oTarget, sListType, sListName); + if (jList == JSON_ARRAY) + return _DeclareList(oTarget, sListType, sListName, nLength, jDefault); + else if (nLength < 0) + return jList; + else + { + int n, nList = JsonGetLength(jList); + if (nList > nLength) + jList = JsonArrayGetRange(jList, 0, nLength - 1); + else + { + for (n = 0; n < nLength - nList; n++) + JsonArrayInsertInplace(jList, jDefault); + } + + _SetList(oTarget, sListType, sListName, jList); + } + + return jList; +} + +// Returns the length of array sListName on oTarget. +int _CountList(object oTarget, string sListType, string sListName) +{ + return JsonGetLength(_GetList(oTarget, sListType, sListName)); +} + +// Sorts sListName on oTarget in order specified by nOrder. +void _SortList(object oTarget, string sListType, string sListName, int nOrder) +{ + json jList = _GetList(oTarget, sListType, sListName); + + if (JsonGetLength(jList) > 1) + _SetList(oTarget, sListType, sListName, JsonArrayTransform(jList, nOrder)); +} + +// ----------------------------------------------------------------------------- +// Public Functions +// ----------------------------------------------------------------------------- + +json VectorToJson(vector vPosition = [0.0, 0.0, 0.0]) +{ + json jPosition = JSON_OBJECT; + JsonObjectSetInplace(jPosition, "x", JsonFloat(vPosition.x)); + JsonObjectSetInplace(jPosition, "y", JsonFloat(vPosition.y)); + JsonObjectSetInplace(jPosition, "z", JsonFloat(vPosition.z)); + + return jPosition; +} + +json JsonVector(vector vPosition = [0.0, 0.0, 0.0]) +{ + return VectorToJson(vPosition); +} + +vector JsonToVector(json jPosition) +{ + float x = JsonGetFloat(JsonObjectGet(jPosition, "x")); + float y = JsonGetFloat(JsonObjectGet(jPosition, "y")); + float z = JsonGetFloat(JsonObjectGet(jPosition, "z")); + + return Vector(x, y, z); +} + +vector JsonGetVector(json jPosition) +{ + return JsonToVector(jPosition); +} + +json LocationToJson(location lLocation) +{ + json jLocation = JSON_OBJECT; + JsonObjectSetInplace(jLocation, "area", JsonString(GetTag(GetAreaFromLocation(lLocation)))); + JsonObjectSetInplace(jLocation, "position", VectorToJson(GetPositionFromLocation(lLocation))); + JsonObjectSetInplace(jLocation, "facing", JsonFloat(GetFacingFromLocation(lLocation))); + + return jLocation; +} + +json JsonLocation(location lLocation) +{ + return LocationToJson(lLocation); +} + +location JsonToLocation(json jLocation) +{ + object oArea = GetObjectByTag(JsonGetString(JsonObjectGet(jLocation, "area"))); + vector vPosition = JsonToVector(JsonObjectGet(jLocation, "position")); + float fFacing = JsonGetFloat(JsonObjectGet(jLocation, "facing")); + + return Location(oArea, vPosition, fFacing); +} + +location JsonGetLocation(json jLocation) +{ + return JsonToLocation(jLocation); +} + +int AddListFloat(object oTarget, float fValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, JsonFloat(fValue), -1, bAddUnique); +} + +int AddListInt(object oTarget, int nValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_INT, sListName, JsonInt(nValue), -1, bAddUnique); +} + +int AddListLocation(object oTarget, location lValue, string sListName = "", int bAddUnique = FALSE) +{ + json jLocation = LocationToJson(lValue); + return _InsertListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, jLocation, -1, bAddUnique); +} + +int AddListVector(object oTarget, vector vValue, string sListName = "", int bAddUnique = FALSE) +{ + json jVector = VectorToJson(vValue); + return _InsertListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, jVector, -1, bAddUnique); +} + +int AddListObject(object oTarget, object oValue, string sListName = "", int bAddUnique = FALSE) +{ + json jObject = JsonString(ObjectToString(oValue)); + return _InsertListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, jObject, -1, bAddUnique); +} + +int AddListString(object oTarget, string sString, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_STRING, sListName, JsonString(sString), -1, bAddUnique); +} + +int AddListJson(object oTarget, json jValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_JSON, sListName, jValue, -1, bAddUnique); +} + +float GetListFloat(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, nIndex); + return jValue == JSON_NULL ? 0.0 : JsonGetFloat(jValue); +} + +int GetListInt(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_INT, sListName, nIndex); + return jValue == JSON_NULL ? -1 : JsonGetInt(jValue); +} + +location GetListLocation(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, nIndex); + + if (jValue == JSON_NULL) + return Location(OBJECT_INVALID, Vector(), 0.0); + else + return JsonToLocation(jValue); +} + +vector GetListVector(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, nIndex); + + if (jValue == JSON_NULL) + return Vector(); + else + return JsonToVector(jValue); +} + +object GetListObject(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, nIndex); + return jValue == JSON_NULL ? OBJECT_INVALID : StringToObject(JsonGetString(jValue)); +} + +string GetListString(object oTarget, int nIndex = 0, string sListName = "") +{ + json jValue = _GetListElement(oTarget, VARLIST_TYPE_STRING, sListName, nIndex); + return jValue == JSON_NULL ? "" : JsonGetString(jValue); +} + +json GetListJson(object oTarget, int nIndex = 0, string sListName = "") +{ + return _GetListElement(oTarget, VARLIST_TYPE_JSON, sListName, nIndex); +} + +int DeleteListFloat(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, nIndex); +} + +int DeleteListInt(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_INT, sListName, nIndex); +} + +int DeleteListLocation(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, nIndex); +} + +int DeleteListVector(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, nIndex); +} + +int DeleteListObject(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, nIndex); +} + +int DeleteListString(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_STRING, sListName, nIndex); +} + +int DeleteListJson(object oTarget, int nIndex, string sListName = "", int bMaintainOrder = FALSE) +{ + return _DeleteListElement(oTarget, VARLIST_TYPE_JSON, sListName, nIndex); +} + +int RemoveListFloat(object oTarget, float fValue, string sListName = "", int bMaintainOrder = FALSE) +{ + return _RemoveListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, JsonFloat(fValue)); +} + +int RemoveListInt(object oTarget, int nValue, string sListName = "", int bMaintainOrder = FALSE) +{ + return _RemoveListElement(oTarget, VARLIST_TYPE_INT, sListName, JsonInt(nValue)); +} + +int RemoveListLocation(object oTarget, location lValue, string sListName = "", int bMaintainOrder = FALSE) +{ + json jLocation = LocationToJson(lValue); + return _RemoveListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, jLocation); +} + +int RemoveListVector(object oTarget, vector vValue, string sListName = "", int bMaintainOrder = FALSE) +{ + json jVector = VectorToJson(vValue); + return _RemoveListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, jVector); +} + +int RemoveListObject(object oTarget, object oValue, string sListName = "", int bMaintainOrder = FALSE) +{ + json jObject = JsonString(ObjectToString(oValue)); + return _RemoveListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, jObject); +} + +int RemoveListString(object oTarget, string sValue, string sListName = "", int bMaintainOrder = FALSE) +{ + return _RemoveListElement(oTarget, VARLIST_TYPE_STRING, sListName, JsonString(sValue)); +} + +int RemoveListJson(object oTarget, json jValue, string sListName = "", int bMaintainOrder = FALSE) +{ + return _RemoveListElement(oTarget, VARLIST_TYPE_JSON, sListName, jValue); +} + +float PopListFloat(object oTarget, string sListName = "") +{ + float f = GetListFloat(oTarget, 0, sListName); + DeleteListFloat(oTarget, 0, sListName); + return f; +} + +int PopListInt(object oTarget, string sListName = "") +{ + int n = GetListInt(oTarget, 0, sListName); + DeleteListInt(oTarget, 0, sListName); + return n; +} + +location PopListLocation(object oTarget, string sListName = "") +{ + location l = GetListLocation(oTarget, 0, sListName); + DeleteListLocation(oTarget, 0, sListName); + return l; +} + +vector PopListVector(object oTarget, string sListName = "") +{ + vector v = GetListVector(oTarget, 0, sListName); + DeleteListVector(oTarget, 0, sListName); + return v; +} + +object PopListObject(object oTarget, string sListName = "") +{ + object o = GetListObject(oTarget, 0, sListName); + DeleteListObject(oTarget, 0, sListName); + return o; +} + +string PopListString(object oTarget, string sListName = "") +{ + string s = GetListString(oTarget, 0, sListName); + DeleteListString(oTarget, 0, sListName); + return s; +} + +json PopListJson(object oTarget, string sListName = "") +{ + json j = GetListJson(oTarget, 0, sListName); + DeleteListString(oTarget, 0, sListName); + return j; +} + +int FindListFloat(object oTarget, float fValue, string sListName = "") +{ + return _FindListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, JsonFloat(fValue)); +} + +int FindListInt(object oTarget, int nValue, string sListName = "") +{ + return _FindListElement(oTarget, VARLIST_TYPE_INT, sListName, JsonInt(nValue)); +} + +int FindListLocation(object oTarget, location lValue, string sListName = "") +{ + json jLocation = LocationToJson(lValue); + return _FindListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, jLocation); +} + +int FindListVector(object oTarget, vector vValue, string sListName = "") +{ + json jVector = VectorToJson(vValue); + return _FindListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, jVector); +} + +int FindListObject(object oTarget, object oValue, string sListName = "") +{ + json jObject = JsonString(ObjectToString(oValue)); + return _FindListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, jObject); +} + +int FindListString(object oTarget, string sValue, string sListName = "") +{ + return _FindListElement(oTarget, VARLIST_TYPE_STRING, sListName, JsonString(sValue)); +} + +int FindListJson(object oTarget, json jValue, string sListName = "") +{ + return _FindListElement(oTarget, VARLIST_TYPE_JSON, sListName, jValue); +} + +int HasListFloat(object oTarget, float fValue, string sListName = "") +{ + return FindListFloat(oTarget, fValue, sListName) != -1; +} + +int HasListInt(object oTarget, int nValue, string sListName = "") +{ + return FindListInt(oTarget, nValue, sListName) != -1; +} + +int HasListLocation(object oTarget, location lValue, string sListName = "") +{ + return FindListLocation(oTarget, lValue, sListName) != -1; +} + +int HasListVector(object oTarget, vector vValue, string sListName = "") +{ + return FindListVector(oTarget, vValue, sListName) != -1; +} + +int HasListObject(object oTarget, object oValue, string sListName = "") +{ + return FindListObject(oTarget, oValue, sListName) != -1; +} + +int HasListString(object oTarget, string sValue, string sListName = "") +{ + return FindListString(oTarget, sValue, sListName) != -1; +} + +int HasListJson(object oTarget, json jValue, string sListName = "") +{ + return FindListJson(oTarget, jValue, sListName) != -1; +} + +int InsertListFloat(object oTarget, int nIndex, float fValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, JsonFloat(fValue), nIndex, bAddUnique); +} + +int InsertListInt(object oTarget, int nIndex, int nValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_INT, sListName, JsonInt(nValue), nIndex, bAddUnique); +} + +int InsertListLocation(object oTarget, int nIndex, location lValue, string sListName = "", int bAddUnique = FALSE) +{ + json jLocation = LocationToJson(lValue); + return _InsertListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, jLocation, nIndex, bAddUnique); +} + +int InsertListVector(object oTarget, int nIndex, vector vValue, string sListName = "", int bAddUnique = FALSE) +{ + json jVector = VectorToJson(vValue); + return _InsertListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, jVector, nIndex, bAddUnique); +} + +int InsertListObject(object oTarget, int nIndex, object oValue, string sListName = "", int bAddUnique = FALSE) +{ + json jObject = JsonString(ObjectToString(oValue)); + return _InsertListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, jObject, nIndex, bAddUnique); +} + +int InsertListString(object oTarget, int nIndex, string sValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_STRING, sListName, JsonString(sValue), nIndex, bAddUnique); +} + +int InsertListJson(object oTarget, int nIndex, json jValue, string sListName = "", int bAddUnique = FALSE) +{ + return _InsertListElement(oTarget, VARLIST_TYPE_JSON, sListName, jValue, nIndex, bAddUnique); +} + +void SetListFloat(object oTarget, int nIndex, float fValue, string sListName = "") +{ + _SetListElement(oTarget, VARLIST_TYPE_FLOAT, sListName, nIndex, JsonFloat(fValue)); +} + +void SetListInt(object oTarget, int nIndex, int nValue, string sListName = "") +{ + _SetListElement(oTarget, VARLIST_TYPE_INT, sListName, nIndex, JsonInt(nValue)); +} + +void SetListLocation(object oTarget, int nIndex, location lValue, string sListName = "") +{ + json jLocation = LocationToJson(lValue); + _SetListElement(oTarget, VARLIST_TYPE_LOCATION, sListName, nIndex, jLocation); +} + +void SetListVector(object oTarget, int nIndex, vector vValue, string sListName = "") +{ + json jVector = VectorToJson(vValue); + _SetListElement(oTarget, VARLIST_TYPE_VECTOR, sListName, nIndex, jVector); +} + +void SetListObject(object oTarget, int nIndex, object oValue, string sListName = "") +{ + json jObject = JsonString(ObjectToString(oValue)); + _SetListElement(oTarget, VARLIST_TYPE_OBJECT, sListName, nIndex, jObject); +} + +void SetListString(object oTarget, int nIndex, string sValue, string sListName = "") +{ + _SetListElement(oTarget, VARLIST_TYPE_STRING, sListName, nIndex, JsonString(sValue)); +} + +void SetListJson(object oTarget, int nIndex, json jValue, string sListName = "") +{ + _SetListElement(oTarget, VARLIST_TYPE_JSON, sListName, nIndex, jValue); +} + +int CopyListFloat(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_FLOAT, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListInt(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_INT, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListLocation(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_LOCATION, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListVector(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_VECTOR, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListObject(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_OBJECT, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListString(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_STRING, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int CopyListJson(object oSource, object oTarget, string sSourceName, string sTargetName, int nIndex, int nRange = 1, int bAddUnique = FALSE) +{ + return _CopyListElements(oSource, oTarget, VARLIST_TYPE_JSON, sSourceName, sTargetName, nIndex, nRange, bAddUnique); +} + +int IncrementListInt(object oTarget, int nIndex, int nIncrement = 1, string sListName = "") +{ + return _IncrementListElement(oTarget, sListName, nIndex, nIncrement); +} + +int DecrementListInt(object oTarget, int nIndex, int nDecrement = -1, string sListName = "") +{ + return _IncrementListElement(oTarget, sListName, nIndex, nDecrement); +} + +json GetFloatList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_FLOAT, sListName); +} + +json GetIntList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_INT, sListName); +} + +json GetLocationList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_LOCATION, sListName); +} + +json GetVectorList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_VECTOR, sListName); +} + +json GetObjectList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_OBJECT, sListName); +} + +json GetStringList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_STRING, sListName); +} + +json GetJsonList(object oTarget, string sListName = "") +{ + return _GetList(oTarget, VARLIST_TYPE_JSON, sListName); +} + +void SetFloatList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_FLOAT, sListName, jList); +} + +void SetIntList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_INT, sListName, jList); +} + +void SetLocationList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_LOCATION, sListName, jList); +} + +void SetVectorList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_VECTOR, sListName, jList); +} + +void SetObjectList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_OBJECT, sListName, jList); +} + +void SetStringList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_STRING, sListName, jList); +} + +void SetJsonList(object oTarget, json jList, string sListName = "") +{ + _SetList(oTarget, VARLIST_TYPE_JSON, sListName, jList); +} + +void DeleteFloatList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_FLOAT, sListName); +} + +void DeleteIntList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_INT, sListName); +} + +void DeleteLocationList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_LOCATION, sListName); +} + +void DeleteVectorList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_VECTOR, sListName); +} + +void DeleteObjectList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_OBJECT, sListName); +} + +void DeleteStringList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_STRING, sListName); +} + +void DeleteJsonList(object oTarget, string sListName = "") +{ + _DeleteList(oTarget, VARLIST_TYPE_JSON, sListName); +} + +json DeclareFloatList(object oTarget, int nCount, string sListName = "", float fDefault = 0.0) +{ + return _DeclareList(oTarget, VARLIST_TYPE_FLOAT, sListName, nCount, JsonFloat(fDefault)); +} + +json DeclareIntList(object oTarget, int nCount, string sListName = "", int nDefault = 0) +{ + return _DeclareList(oTarget, VARLIST_TYPE_INT, sListName, nCount, JsonInt(nDefault)); +} + +json DeclareLocationList(object oTarget, int nCount, string sListName = "") +{ + return _DeclareList(oTarget, VARLIST_TYPE_LOCATION, sListName, nCount, JSON_NULL); +} + +json DeclareVectorList(object oTarget, int nCount, string sListName = "") +{ + return _DeclareList(oTarget, VARLIST_TYPE_VECTOR, sListName, nCount, JSON_NULL); +} + +json DeclareObjectList(object oTarget, int nCount, string sListName = "") +{ + return _DeclareList(oTarget, VARLIST_TYPE_OBJECT, sListName, nCount, JSON_NULL); +} + +json DeclareStringList(object oTarget, int nCount, string sListName = "", string sDefault = "") +{ + return _DeclareList(oTarget, VARLIST_TYPE_STRING, sListName, nCount, JsonString(sDefault)); +} + +json DeclareJsonList(object oTarget, int nCount, string sListName = "") +{ + return _DeclareList(oTarget, VARLIST_TYPE_JSON, sListName, nCount, JSON_NULL); +} + +json NormalizeFloatList(object oTarget, int nCount, string sListName = "", float fDefault = 0.0) +{ + return _NormalizeList(oTarget, VARLIST_TYPE_FLOAT, sListName, nCount, JsonFloat(fDefault)); +} + +json NormalizeIntList(object oTarget, int nCount, string sListName = "", int nDefault = 0) +{ + return _NormalizeList(oTarget, VARLIST_TYPE_INT, sListName, nCount, JsonInt(nDefault)); +} + +json NormalizeLocationList(object oTarget, int nCount, string sListName = "") +{ + return _NormalizeList(oTarget, VARLIST_TYPE_LOCATION, sListName, nCount, JSON_NULL); +} + +json NormalizeVectorList(object oTarget, int nCount, string sListName = "") +{ + return _NormalizeList(oTarget, VARLIST_TYPE_VECTOR, sListName, nCount, JSON_NULL); +} + +json NormalizeObjectList(object oTarget, int nCount, string sListName = "") +{ + return _NormalizeList(oTarget, VARLIST_TYPE_OBJECT, sListName, nCount, JSON_NULL); +} + +json NormalizeStringList(object oTarget, int nCount, string sListName = "", string sDefault = "") +{ + return _NormalizeList(oTarget, VARLIST_TYPE_STRING, sListName, nCount, JsonString(sDefault)); +} + +json NormalizeJsonList(object oTarget, int nCount, string sListName = "") +{ + return _NormalizeList(oTarget, VARLIST_TYPE_JSON, sListName, nCount, JSON_NULL); +} + +void CopyFloatList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_FLOAT, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyIntList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_INT, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyLocationList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_LOCATION, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyVectorList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_VECTOR, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyObjectList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_OBJECT, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyStringList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_STRING, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +void CopyJsonList(object oSource, object oTarget, string sSourceName, string sTargetName, int bAddUnique = FALSE) +{ + _CopyListElements(oSource, oTarget, VARLIST_TYPE_JSON, sSourceName, sTargetName, 0, -1, bAddUnique); +} + +int CountFloatList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_FLOAT, sListName); +} + +int CountIntList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_INT, sListName); +} + +int CountLocationList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_LOCATION, sListName); +} + +int CountVectorList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_VECTOR, sListName); +} + +int CountObjectList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_OBJECT, sListName); +} + +int CountStringList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_STRING, sListName); +} + +int CountJsonList(object oTarget, string sListName = "") +{ + return _CountList(oTarget, VARLIST_TYPE_JSON, sListName); +} + +void SortFloatList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_FLOAT, sListName, nOrder); +} + +void SortIntList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_INT, sListName, nOrder); +} + +void SortStringList(object oTarget, int nOrder = LIST_SORT_ASC, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_STRING, sListName, nOrder); +} + +void ShuffleFloatList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_FLOAT, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleIntList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_INT, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleLocationList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_LOCATION, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleVectorList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_VECTOR, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleObjectList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_OBJECT, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleStringList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_STRING, sListName, JSON_ARRAY_SHUFFLE); +} + +void ShuffleJsonList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_JSON, sListName, JSON_ARRAY_SHUFFLE); +} + +void ReverseFloatList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_FLOAT, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseIntList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_INT, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseLocationList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_LOCATION, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseVectorList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_VECTOR, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseObjectList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_OBJECT, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseStringList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_STRING, sListName, JSON_ARRAY_REVERSE); +} + +void ReverseJsonList(object oTarget, string sListName = "") +{ + _SortList(oTarget, VARLIST_TYPE_JSON, sListName, JSON_ARRAY_REVERSE); +} diff --git a/_module/nss/xx_cb_enter.nss b/_module/nss/xx_cb_enter.nss index d854a3f..d177971 100644 --- a/_module/nss/xx_cb_enter.nss +++ b/_module/nss/xx_cb_enter.nss @@ -12,6 +12,7 @@ void main () AddJournalQuestEntry("xprules", 1, oPlayer, FALSE, FALSE, FALSE); AddJournalQuestEntry("lvl_adj", 1, oPlayer, FALSE, FALSE, FALSE); + AddJournalQuestEntry("JRNL_PRC8", 1, oPlayer, FALSE, FALSE, FALSE); cs_DestroyStolenItems(oPlayer); // Destroy any stolen items cs_LimitGoldInBank(oPlayer, 20000000); // Limit gold in bank diff --git a/_module/utd/inn2pcrooms.utd.json b/_module/utd/inn2pcrooms.utd.json new file mode 100644 index 0000000..894e905 --- /dev/null +++ b/_module/utd/inn2pcrooms.utd.json @@ -0,0 +1,213 @@ +{ + "__data_type": "UTD ", + "AnimationState": { + "type": "byte", + "value": 0 + }, + "Appearance": { + "type": "dword", + "value": 0 + }, + "AutoRemoveKey": { + "type": "byte", + "value": 0 + }, + "CloseLockDC": { + "type": "byte", + "value": 0 + }, + "Comment": { + "type": "cexostring", + "value": "\"Create Area\" Door" + }, + "Conversation": { + "type": "resref", + "value": "doorcv_inn_room" + }, + "CurrentHP": { + "type": "short", + "value": 15 + }, + "Description": { + "type": "cexolocstring", + "value": { + "0": "This door leads to the inn rooms above.", + "id": 9078 + } + }, + "DisarmDC": { + "type": "byte", + "value": 15 + }, + "Faction": { + "type": "dword", + "value": 1 + }, + "Fort": { + "type": "byte", + "value": 16 + }, + "GenericType_New": { + "type": "dword", + "value": 12 + }, + "Hardness": { + "type": "byte", + "value": 5 + }, + "HP": { + "type": "short", + "value": 15 + }, + "Interruptable": { + "type": "byte", + "value": 1 + }, + "KeyName": { + "type": "cexostring", + "value": "" + }, + "KeyRequired": { + "type": "byte", + "value": 1 + }, + "LinkedTo": { + "type": "cexostring", + "value": "" + }, + "LinkedToFlags": { + "type": "byte", + "value": 0 + }, + "LoadScreenID": { + "type": "word", + "value": 0 + }, + "Lockable": { + "type": "byte", + "value": 0 + }, + "Locked": { + "type": "byte", + "value": 1 + }, + "LocName": { + "type": "cexolocstring", + "value": { + "0": "\"Create Area\" Door", + "id": 5349 + } + }, + "OnClick": { + "type": "resref", + "value": "" + }, + "OnClosed": { + "type": "resref", + "value": "" + }, + "OnDamaged": { + "type": "resref", + "value": "" + }, + "OnDeath": { + "type": "resref", + "value": "x2_door_death" + }, + "OnDisarm": { + "type": "resref", + "value": "" + }, + "OnFailToOpen": { + "type": "resref", + "value": "cv_inn_door" + }, + "OnHeartbeat": { + "type": "resref", + "value": "" + }, + "OnLock": { + "type": "resref", + "value": "" + }, + "OnMeleeAttacked": { + "type": "resref", + "value": "" + }, + "OnOpen": { + "type": "resref", + "value": "" + }, + "OnSpellCastAt": { + "type": "resref", + "value": "" + }, + "OnTrapTriggered": { + "type": "resref", + "value": "" + }, + "OnUnlock": { + "type": "resref", + "value": "" + }, + "OnUserDefined": { + "type": "resref", + "value": "" + }, + "OpenLockDC": { + "type": "byte", + "value": 18 + }, + "PaletteID": { + "type": "byte", + "value": 0 + }, + "Plot": { + "type": "byte", + "value": 1 + }, + "PortraitId": { + "type": "word", + "value": 2624 + }, + "Ref": { + "type": "byte", + "value": 0 + }, + "Tag": { + "type": "cexostring", + "value": "inn2pcrooms" + }, + "TemplateResRef": { + "type": "resref", + "value": "inn2pcrooms" + }, + "TrapDetectable": { + "type": "byte", + "value": 1 + }, + "TrapDetectDC": { + "type": "byte", + "value": 0 + }, + "TrapDisarmable": { + "type": "byte", + "value": 1 + }, + "TrapFlag": { + "type": "byte", + "value": 0 + }, + "TrapOneShot": { + "type": "byte", + "value": 1 + }, + "TrapType": { + "type": "byte", + "value": 0 + }, + "Will": { + "type": "byte", + "value": 0 + } +} diff --git a/_module/uti/ringoflight.uti.json b/_module/uti/ringoflight.uti.json index 6dfc48b..912fc9a 100644 --- a/_module/uti/ringoflight.uti.json +++ b/_module/uti/ringoflight.uti.json @@ -20,6 +20,10 @@ "type": "dword", "value": 351751 }, + "Cursed": { + "type": "byte", + "value": 0 + }, "DescIdentified": { "type": "cexolocstring", "value": {} @@ -256,5 +260,9 @@ "TemplateResRef": { "type": "resref", "value": "ringoflight" + }, + "xModelPart1": { + "type": "word", + "value": 13 } } diff --git a/_release/Realms of Trinity II [PRC].7z b/_release/Realms of Trinity II [PRC].7z index dbc0d73..47bb34b 100644 Binary files a/_release/Realms of Trinity II [PRC].7z and b/_release/Realms of Trinity II [PRC].7z differ